Версия для печати темы
Нажмите сюда для просмотра этой темы в оригинальном формате
Форум программистов > C/C++: Системное программирование и WinAPI > Буфер обмена в формате CF_BITMAP


Автор: dimbas 9.10.2010, 21:29
Необходимо отрисовать на форме изображение из буфера обмена. Есть обработчик:
Код

    case WM_PAINT:
      hdc = BeginPaint(hWnd, &ps);
      
      //if(IsClipboardFormatAvailable(CF_BITMAP))
      {
          HBITMAP hbmClipData;
          HDC hdcBitmap;
          HDC dc;

          // Получаем содержимое буфера
          hbmClipData = (HBITMAP)GetClipboardData(CF_BITMAP);
          //hbmClipData = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_BITMAP1));
          if (!hbmClipData)
            MessageBox(hWnd, L"GetClipboardData(CF_BITMAP) has failed",L"Failed", MB_OK);
          
          hdcBitmap = CreateCompatibleDC(GetDC(hWnd));
          SelectObject(hdcBitmap, hbmClipData);

          if(!BitBlt(hdc,
                     0,0, 
                     48, 48, 
                     hdcBitmap,
                     0,0,
                     SRCCOPY))
            MessageBox(hWnd, L"BitBlt has failed",L"Failed", MB_OK);

        EndPaint(hWnd, &ps);
      }
      break;



Говорит, что "GetClipboardData(CF_BITMAP) has failed" =)
Если GetClipboardData закомментировать и ,соответственно, раскомментировать LoadImage, все работает прекрасно, то есть, рисуется бмп из ресурса. В чем может быть проблема? Не конвертится HANDLE к HBITMAP-у?

Автор: xvr 10.10.2010, 09:58
Цитата(dimbas @  9.10.2010,  21:29 Найти цитируемый пост)
В чем может быть проблема?
Видимо в том, что в Clipboard лежит не Bitmap. Clipboard не может вернуть любой формат, который захочется программисту, а только тот, который в нее положили.
Точнее некоторые форматы она может конвертировать (см http://msdn.microsoft.com/en-us/library/ms649013%28v=VS.85%29.aspx#_win32_Synthesized_Clipboard_Formats ), но далеко не все и не во все

PS. Надеюсь OpenClipboard вы не забыли позвать?

Автор: dimbas 10.10.2010, 12:33
Насчет форматов это понятно) Что перед запуском проги, что во время работы копирую кусок из пейнта. Дело не в этом.

Только хотел писать "во я дурак, забыл openclipboard", но не тут то было. Читается из клипбоарда и рисуется, только исчезает после перерисовки. То есть, отрисовывается только тогда, когда вызывается InvalidateRect() по событию WM_DRAWCLIPBOARD. Урезанный код WndProc:
Код

LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
  static HWND hNextViewer;
  int wmId, wmEvent;
  PAINTSTRUCT ps;
  static HDC hdc;
  
  // регистрируемся как наблюдатель за буфером обмена
  if (!bSpy) // только один раз. bSpy глобально объявлен и на старте = 0
  {
    hNextViewer = SetClipboardViewer(hWnd);
    bSpy = 1;
  }

  switch (message)
  {
    /* ............................... */

    case WM_PAINT:
      hdc = BeginPaint(hWnd, &ps);
      
          if (!OpenClipboard(hWnd))
            MessageBox(hWnd, L"OpenClipboard(hWnd) has failed",L"Failed", MB_OK);

          HBITMAP hbmClipData;
          HDC hdcBitmap;
          HDC dc;

          // Получаем содержимое буфера
          hbmClipData = (HBITMAP)GetClipboardData(CF_BITMAP);
          //hbmClipData = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_BITMAP1));
          if (!hbmClipData)
            MessageBox(hWnd, L"GetClipboardData(CF_BITMAP) has failed",L"Failed", MB_OK);
          
          // Retrieve the handle to a display device context for the client area of the window. 
          hdcBitmap = CreateCompatibleDC(GetDC(hWnd));
          // назначаем
          SelectObject(hdcBitmap, hbmClipData);

          if(!BitBlt(hdc,
                     0,0,
                     48, 48, 
                     hdcBitmap,
                     0,0,
                     SRCCOPY))
            MessageBox(hWnd, L"BitBlt has failed",L"Failed", MB_OK);

          CloseClipboard();

      EndPaint(hWnd, &ps);
      break;

    case WM_DRAWCLIPBOARD:
      
      // игнорирование других форматов
      if(IsClipboardFormatAvailable(CF_BITMAP))
        InvalidateRect(hWnd, NULL, TRUE);

      // передаем сообщение следующему наблюдателю
            if(IsWindow(hNextViewer))
                PostMessage(hNextViewer, message, wParam, lParam);

      break;

    // обработка сообщения изменения очереди наблюдателей
    case WM_CHANGECBCHAIN:
            if(hNextViewer == (HWND)wParam)
                hNextViewer = (HWND)lParam;
            else if(IsWindow(hNextViewer))
                PostMessage(hNextViewer, message, wParam, lParam);
      return TRUE;
      break;

    case WM_DESTROY:
      // отписываемся от наблюдения
      ChangeClipboardChain(hWnd, hNextViewer);
      PostQuitMessage(0);
      break;

    default:
      return DefWindowProc(hWnd, message, wParam, lParam);
  }
  return 0;
}


Можно было бы подумать, что какие-то проблемы со слежением за буфером. Но ведь как раз после Ctrl+C-то и рисуется (я открываю свою прогу слева, Пейнт справа, так чтобы он не перекрывал те 48х48, где происходит рисование, и копирую из пейнта ровно 48х48 для уверенности. Если "испортить" часть отрисованного, оно остается белым). Проверка на формат работает: если скопировать текст, прога ругается "GetClipboardData(CF_BITMAP)". Наверное, я где-то опять туплю
ПС. а Опенклипбоард все-таки забыл))

upd
Если читать из ресурса, то изображение остается на форме после перерисовки!

upd2
Если закомментировать InvalidateRect() в case WM_DRAWCLIPBOARD и перед запуском скопировать в буфер картинку, то она отобразится сразу, но больше никогда

Автор: xvr 10.10.2010, 18:09
Видимо кто то обнуляет clipboard после первой же перерисовки (проверьте, остается ли содержимое clipboard'а)
Еще подозрительно выглядит отсутствие SelectObject на первоначальный объект после перерисовки (хотя может быть это и пройдет - надо проверить)

Автор: dimbas 10.10.2010, 18:32
Цитата(xvr @ 10.10.2010,  18:09)

Видимо кто то обнуляет clipboard после первой же перерисовки (проверьте, остается ли содержимое clipboard'а)

Проверил вставкой в Пейнт после перерисовки. Не обнуляет.

Цитата(xvr @ 10.10.2010,  18:09)

Еще подозрительно выглядит отсутствие SelectObject на первоначальный объект после перерисовки (хотя может быть это и пройдет - надо проверить)

Простите, не понял немного))

Автор: xvr 10.10.2010, 22:37
Цитата(dimbas @  10.10.2010,  18:32 Найти цитируемый пост)
Простите, не понял немного)) 
Из MSDN:
Цитата

Remarks

This function [SelectObject] returns the previously selected object of the specified type. An application should always replace a new object with the original, default object after it has finished drawing with the new object.


Автор: dimbas 11.10.2010, 10:30
Код

LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
  /*...*/
  HGDIOBJ def;
  /*...*/
  case WM_PAINT:
  /*...*/
  def = SelectObject(hdcBitmap, hbmClipData);
  /*...*/
  EndPaint(hWnd, &ps);
  SelectObject(hdcBitmap, def);
  /*...*/


Спасибо! Заработало!))  smile 

Powered by Invision Power Board (http://www.invisionboard.com)
© Invision Power Services (http://www.invisionpower.com)