DirectDrawCreate(0, &ddraw1, 0);
 ddraw1->QueryInterface(IID_IDirectDraw2, (void**)&ddraw2);  ddraw1->Release(), ddraw1=0;  ddraw2->SetCooperativeLevel (GetSafeHwnd(), DDSCL_NORMAL);
 DetectDisplayMode();
 if (CreateFlippingSurfaces()==FALSE) {
  AfxMessageBox('CreateFlippingSurfaces() failed');
  return FALSE;
 }
 if (CreateCustomSurfaces()==FALSE) {
  AfxMessageBox('CreateCustomSurfaces() failed');
  return FALSE;
 }
 return 0;
}
Сначала указатель на интерфейс DirectDraw(ddraw1) инициализируется функцией DirectDrawCreate(). Указатель ddraw1, как и в полноэкранной версии, используется только для получения указателя на интерфейс DirectDraw2, после чего освобождается.
Затем функция OnCreate() вызывает функцию SetCooperativeLevel(). В полноэкранном приложении уровень кооперации определялся тремя флагами: DDSCL_EXCLUSIVE, DDSCL_FULLSCREEN и DDSCL_ALLOWMODEX. В данном случае используется только флаг DDSCL_NORMAL.
Функция DetectDisplayMode() инициализирует некоторые переменные класса DirectDrawWin. Она выглядит так:
BOOL DirectDrawWin::DetectDisplayMode() {
 DDSURFACEDESC desc;
 ZeroMemory(&desc, sizeof(desc));
 desc.dwSize=sizeof(desc);
 if (ddraw2->GetDisplayMode(&desc)!=DD_OK) {
  TRACE('GetDisplayMode() failed
');
  return FALSE;
 }
 displayrect.left=0;
 displayrect.top=0;
 displayrect.right=desc.dwWidth;
 displayrect.bottom=desc.dwHeight;
 displaydepth=desc.ddpfPixelFormat.dwRGBBitCount;
 return TRUE;
}
Функция DetectDisplayMode() с помощью функции GetDisplayMode() интерфейса DirectDraw получает информацию о текущем видеорежиме Windows. Говоря точнее, разрешение экрана и глубина пикселей текущего видеорежима сохраняются в переменных displayrect и displaydepth.
Далее OnCreate() вызывает функцию CreateFlippingSurfaces(). Хотя оконное приложение не может выполнять настоящего переключения страниц (как можно было бы решить, исходя из имени функции), имя было сохранено, потому что создаваемые в ней поверхности эмулируют переключение страниц. Код функции приведен в листинге 3.4.
Листинг 3.4. Функция CreateFlippingSurfaces() в оконном приложении
BOOL DirectDrawWin::CreateFlippingSurfaces() {
 HRESULT r;
 DDSURFACEDESC desc;
 desc.dwSize = sizeof(desc);
 desc.dwFlags = DDSD_CAPS;
 desc.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;
 r=ddraw2->CreateSurface(&desc, &primsurf, 0);
 if (r!=DD_OK) {
  TRACE('FAILED to create 'primsurf'
');
  return FALSE;
 }
 r=ddraw2->CreateClipper(0, &clipper, 0);
 if (r!=DD_OK) {
  TRACE('CreateClipper() failed
');
  return FALSE;
 }
 r=clipper->SetHWnd(0, GetSafeHwnd());
 if (r!=DD_OK) {
  TRACE('SetHWnd() failed
');
  return FALSE;
 }
 r=primsurf->SetClipper(clipper);
 if (r!=DD_OK) {
  TRACE('SetClipper() failed
');
  return FALSE;
 }
 ZeroMemory(&desc, sizeof(desc));
 desc.dwSize = sizeof(desc);
 desc.dwFlags = DDSD_WIDTH | DDSD_HEIGHT | DDSD_CAPS;
 desc.dwWidth = displayrect.Width();
 desc.dwHeight = displayrect.Height();
 desc.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN | DDSCAPS_VIDEOMEMORY;
 r=ddraw2->CreateSurface(&desc, &backsurf, 0);
 if (r!=DD_OK) {
  TRACE('failed to create 'backsurf' in video
');
  videobacksurf=FALSE;
 } else {
  TRACE('Created backsurf in video
');
  videobacksurf=TRUE;
 }
 return TRUE;
}
Сначала мы создаем первичную поверхность. В полноэкранном варианте код выглядит по-другому, потому что здесь создается обычная, несоставная первичная поверхность. В структуре DDSURFACEDESC мы описываем первичную поверхность, используя только флаг DDSCAPS_PRIMARYSURFACE. Затем описанная поверхность создается функцией CreateSurface() интерфейса DirectDraw.
Далее функция CreateClipper() интерфейса DirectDraw создает объект отсечения. CreateClipper() получает три аргумента, однако первый и последний из них чаще всего равны нулю. Второй аргумент представляет собой адрес указателя на интерфейс DirectDrawClipper. В нашем случае используется переменная класса DirectDrawWin с именем clipper.
Объект отсечения нужен для ограничения вывода в программе. Поскольку наше приложение работает в окне, которое находится на рабочем столе вместе с другими окнами, при обновлении изображения необходимо учитывать присутствие этих окон. Чтобы объект отсечения автоматически выполнял свою работу, его необходимо присоединить к окну функцией SetHWnd() интерфейса DirectDrawClipper. Функция SetHWnd() получает два аргумента — двойное слово (DWORD), которое зарезервировано для будущего использования и
