Johnny_Depp18 часов назад
Direct 2D #11. Анимации
Уровень сложностиПростойВремя на прочтение12 минОхват и читатели7.3KWindows*C++*Разработка игр*ТуториалВсем здравствуйте! Появилось время, и сразу пишу вам. Сегодня поговорим об анимации.Важно понимать, что Direct2D - это низкоуровневое API, и готовых инструментов в нём нет. Однако существуют вспомогательные компоненты: Windows Animation Manager (WAM) и DirectComposition. Первый появился в Windows 7, второй - в Windows 8.
Windows Animation Manager создан для того, чтобы избавить разработчика от математических расчётов анимации: линейной интерполяции, кривых ускорения/замедления, эффекта пружины и т. д. Вы просто задаёте системе начальные и конечные координаты и эффект, а WAM самостоятельно вычисляет.
Ключевые понятия WAM:
IUIAnimationManager - главный управляющий объект. Он создаёт анимации, отслеживает их выполнение и уведомляет о необходимости обновления кадра.
IUIAnimationVariable - анимируемая переменная. Это может быть координата X, прозрачность, масштаб или любое другое число с плавающей запятой. Вы просто устанавливаете текущее значение переменной, например: «эта переменная сейчас равна 100».
IUIAnimationTransition - закон изменения переменной во времени. WAM предоставляет богатую библиотеку встроенных переходов: линейный (CreateLinearTransition) , с ускорением и замедлением (CreateAccelerateDecelerateTransition), "пружина" (CreateSpringTransition) и многие другие(IUIAnimationTransitionLibraryи IUIAnimationTransitionLibrary2).
IUIAnimationStoryboard - "раскадровка", контейнер, в который вы собираете один или несколько переходов и запускаете их одновременно или последовательно.
В качестве краткого примера (к 11-й статье вы уже умеете работать с графикой, и нет нужды каждый раз писать полотно текста) приведём фрагмент:
В классе окна или приложения объявим необходимые COM-указатели:
// WAM
CComPtr<IUIAnimationManager> pAnimManager;
CComPtr<IUIAnimationTransitionLibrary> pTransitionLibrary;
CComPtr<IUIAnimationVariable> pAnimVarX;
CComPtr<IUIAnimationStoryboard> pStoryboard;
// Direct2D (упрощённо)
CComPtr<ID2D1Factory> pD2DFactory;
CComPtr<ID2D1HwndRenderTarget> pRenderTarget;
CComPtr<ID2D1SolidColorBrush> pBrush;При старте приложения (например, в обработчике WM_CREATE) инициализируется COM и создаются необходимые объекты.
// Инициализация COM (если ещё не сделано) CoInitialize(NULL);
// Создаём менеджер анимаций
CoCreateInstance(CLSID_UIAnimationManager, NULL, CLSCTX_INPROC_SERVER,
IID_IUIAnimationManager, (void**)&pAnimManager);
// Создаём библиотеку переходов
CoCreateInstance(CLSID_UIAnimationTransitionLibrary, NULL, CLSCTX_INPROC_SERVER,
IID_IUIAnimationTransitionLibrary, (void**)&pTransitionLibrary);
// Создаём переменную для координаты X. Начальное значение = 0
pAnimManager->CreateAnimationVariable(0.0, &pAnimVarX);Ниже приведены пояснения к аргументам используемых функций.
CoInitialize:
Первый аргумент зарезервирован и должен принимать значение NULL.
CoCreateInstance:
Первый аргумент - идентификатор класса (CLSID).
Второй аргумент - указатель на управляющий IUnknown при агрегации объектов; в нашем случае агрегация не используется, поэтому передаётся NULL.
Третий аргумент - контекст выполнения. Значение CLSCTX_INPROC_SERVER указывает, что объект должен быть загружен как DLL внутри текущего процесса.
Четвёртый аргумент - указатель на требуемый интерфейс (IID).
Пятый аргумент - адрес переменной, в которую будет передан указатель на созданный объект.
CreateAnimationVariable:
Первый аргумент - начальное значение переменной (тип DOUBLE).
Второй аргумент - адрес указателя на объект IUIAnimationVariable, который будет создан.
Теперь предположим, что мы хотим анимировать перемещение прямоугольника из позиции X = 0 в X = 300 за 2 секунды с эффектом ускорения и замедления.
// Создаём переход: ускорение-замедление, длительность 2 сек, конечное значение 300
CComPtr<IUIAnimationTransition> pTransition;
pTransitionLibrary->CreateAccelerateDecelerateTransition(
2.0, // длительность в секундах
300.0, // конечное значение
0.3, // доля ускорения (0..1)
0.3, // доля замедления (0..1)
&pTransition
);
// Создаём раскадровку pAnimManager->CreateStoryboard(&pStoryboard);
// Добавляем переход к переменной X pStoryboard->AddTransition(pAnimVarX, pTransition);
// Запускаем раскадровку (сразу)
pStoryboard->Schedule(0); // 0 = запустить немедленноПояснение аргументов методов:
CreateAccelerateDecelerateTransition- уже описан в комментариях выше (параметры: ускорение, замедление и продолжительность).
CreateStoryboard(метод IUIAnimationManager):
Единственный аргумент - адрес указателя на создаваемую раскадровку (IUIAnimationStoryboard*). Раскадровка представляет собой контейнер, который может содержать один или несколько переходов, применяемых к разным переменным. Все переходы внутри одной раскадровки запускаются одновременно (если не заданы индивидуальные задержки).
AddTransition:
Первый аргумент - указатель на анимируемую переменную (IUIAnimationVariable*).
Второй аргумент - указатель на ранее созданный переход (IUIAnimationTransition*). Именно этот переход определяет закон изменения переменной во времени.
Schedule:
Первый аргумент - момент времени (в секундах) относительно текущего времени системы, с которого должна запуститься раскадровка. Обычно передаётся 0.0 для немедленного старта.
После вызова Schedule анимация начинает выполняться. WAM автоматически вычисляет промежуточные значения переменной в зависимости от текущего времени.
В каждом кадре (например, в обработчике WM_PAINT или в отдельном потоке рендеринга) необходимо выполнить следующие шаги:
Обновить состояние WAM, передав прошедшее время (с помощью метода Update менеджера).
Получить текущее значение анимируемой переменной (метод GetValue).
void RenderFrame()
{
// 1. Получить время, прошедшее с прошлого кадра (в секундах)
static double lastTime = 0.0;
double currentTime = GetCurrentTimeInSeconds(); // ваша функция получения времени
double deltaTime = currentTime - lastTime;
lastTime = currentTime;
// 2. Обновить WAM. Передаём дельту времени (в секундах)
HRESULT hr = pAnimManager->Update(deltaTime);
// Если анимация завершилась, можно обработать и, например, запустить обратную
// 3. Получить текущее значение X double currentX; pAnimVarX->GetValue(¤tX);
// 4. Начать рисование через Direct2D
pRenderTarget->BeginDraw();
pRenderTarget->Clear(D2D1::ColorF(D2D1::ColorF::White));
// Рисуем прямоугольник размером 50x50 в точке (currentX, 50)
D2D1_RECT_F rect = D2D1::RectF((FLOAT)currentX, 50.0f, (FLOAT)(currentX + 50), 100.0f);
pRenderTarget->FillRectangle(&rect, pBrush);
// Завершаем рисование
hr = pRenderTarget->EndDraw();
if (hr == D2DERR_RECREATE_TARGET) { /* восстановить ресурсы */ }
}Чтобы организовать движение прямоугольника туда-обратно, можно подписаться на событие завершения раскадровки. Однако для простоты реализуем циклическую проверку: в каждом кадре или в отдельном цикле отслеживаем состояние анимации, и если она завершена, запускаем новую раскадровку с обратным направлением движения.
// После pAnimManager->Update(deltaTime) можно проверить статус:
UI_ANIMATION_MANAGER_STATUS status;
pAnimManager->GetStatus(&status);
if (status == UI_ANIMATION_MANAGER_IDLE)
{
// Анимация завершена - запускаем новую в обратном направлении
// Получаем текущее значение X (например, 300)
double currentX;
pAnimVarX->GetValue(¤tX);
// Создаём переход к 0 (или в зависимости от текущего положения)
CComPtr<IUIAnimationTransition> pReverseTransition;
pTransitionLibrary->CreateAccelerateDecelerateTransition(
2.0, 0.0, 0.3, 0.3, &pReverseTransition
);
pAnimManager->CreateStoryboard(&pStoryboard);
pStoryboard->AddTransition(pAnimVarX, pReverseTransition);
pStoryboard->Schedule(0);
}Пояснение аргументов:GetStatusпринимает один аргумент - выходной указатель, в который записывается текущее состояние менеджера. Возможные значения:
UI_ANIMATION_MANAGER_IDLE - все анимации завершены, менеджер находится в состоянии ожидания;
UI_ANIMATION_MANAGER_BUSY - выполняется как минимум одна раскадровка;
UI_ANIMATION_MANAGER_INSUFFICIENT_PRIORITY - встречается крайне редко и сигнализирует о том, что запланированные анимации не могут быть выполнены из-за конфликтов приоритетов.
Обратите внимание: в вызов Update необходимо передавать актуальное время в секундах. Для его измерения с высокой точностью используйте функцию QueryPerformanceCounter.
double GetCurrentTimeInSeconds()
{
static LARGE_INTEGER frequency = {0};
if (frequency.QuadPart == 0)
QueryPerformanceFrequency(&frequency);
LARGE_INTEGER now;
QueryPerformanceCounter(&now);
return (double)now.QuadPart / (double)frequency.QuadPart;
}При завершении приложения необходимо освободить все COM-объекты (при использовании умных указателей освобождение произойдёт автоматически при выходе за область видимости). Windows Animation Manager не требует явного вызова завершения или освобождения ресурсов.
DirectComposition. Теперь - о нём.
Если WAM работает на центральном процессоре и лишь вычисляет промежуточные значения, то DirectComposition (доступный начиная с Windows 8) функционирует на стороне видеокарты в отдельном потоке. Он принимает готовые растровые изображения (битмапы), применяет к ним трансформации, эффекты и анимации, после чего выводит результирующую сцену на экран с высокой частотой кадров.
Ключевые понятия DirectComposition:
Визуальное дерево (Visual Tree) - иерархия объектов IDCompositionVisual. Корневой визуальный элемент привязан к окну, дочерние позиционируются относительно родителя. Это позволяет легко строить сложные сцены: например, персонаж представляет собой родительский элемент, а его руки и ноги - дочерние визуалы.
Свойства визуала - положение (OffsetX, OffsetY), трансформации (поворот, масштаб), прозрачность и эффекты.
IDCompositionAnimation - анимационная кривая, которую можно применить к любому свойству. Она задаётся в виде сегментов: линейных, кубических, синусоидальных или повторяющихся.
Commit - после настройки дерева и анимаций вызывается метод Commit, после чего все изменения применяются.
Перейдём к примеру.
// Direct3D 11 (нужен для DirectComposition)
CComPtr<ID3D11Device> m_pD3D11Device;
// DirectComposition
CComPtr<IDCompositionDevice> m_pDCompDevice;
CComPtr<IDCompositionTarget> m_pDCompTarget;
CComPtr<IDCompositionVisual> m_pVisual;
// Direct2D (для создания контента)
CComPtr<ID2D1Factory> m_pD2DFactory;
CComPtr<ID2D1Device> m_pD2DDevice;
CComPtr<ID2D1DeviceContext> m_pD2DContext;
CComPtr<ID2D1Bitmap1> m_pD2DBitmap;
CComPtr<ID2D1SolidColorBrush> m_pBrush;
// Объекты анимации
CComPtr<IDCompositionAnimation> m_pAnimX;
CComPtr<IDCompositionAnimation> m_pAnimOpacity;Инициализация. Данный шаг выполняется однократно при запуске приложения (например, в обработчике сообщения WM_CREATE).
HRESULT Init(HWND hwnd) { HRESULT hr = S_OK;
// 1. Создаём устройство Direct3D 11 с поддержкой BGRA (нужно для D2D)
D3D_FEATURE_LEVEL featureLevel;
hr = D3D11CreateDevice(
nullptr,
D3D_DRIVER_TYPE_HARDWARE,
nullptr,
D3D11_CREATE_DEVICE_BGRA_SUPPORT, // Важно для Direct2D!
nullptr,
0,
D3D11_SDK_VERSION,
&m_pD3D11Device,
&featureLevel,
nullptr
);
// 2. Получаем интерфейс IDXGIDevice от устройства D3D
CComPtr<IDXGIDevice> pDXGIDevice;
if (SUCCEEDED(hr))
hr = m_pD3D11Device->QueryInterface(&pDXGIDevice);
// 3. Создаём устройство DirectComposition[reference:1]
if (SUCCEEDED(hr))
hr = DCompositionCreateDevice(pDXGIDevice, __uuidof(IDCompositionDevice),
reinterpret_cast<void**>(&m_pDCompDevice));
// 4. Создаём цель композиции для нашего окна[reference:2]
if (SUCCEEDED(hr))
hr = m_pDCompDevice->CreateTargetForHwnd(hwnd, TRUE, &m_pDCompTarget);
// 5. Создаём фабрику Direct2D
if (SUCCEEDED(hr))
hr = D2D1CreateFactory(D2D1_FACTORY_TYPE_SINGLE_THREADED, &m_pD2DFactory);
// 6. Создаём устройство Direct2D из устройства DXGI
if (SUCCEEDED(hr))
hr = m_pD2DFactory->CreateDevice(pDXGIDevice, &m_pD2DDevice);
// 7. Создаём контекст устройства Direct2D
if (SUCCEEDED(hr))
hr = m_pD2DDevice->CreateDeviceContext(
D2D1_DEVICE_CONTEXT_OPTIONS_NONE,
&m_pD2DContext
);
// 8. Создаём битмап с контентом (квадрат 200x200)
if (SUCCEEDED(hr))
hr = CreateBitmapContent();
// 9. Создаём визуал и устанавливаем ему контент[reference:3]
if (SUCCEEDED(hr))
hr = m_pDCompDevice->CreateVisual(&m_pVisual);
if (SUCCEEDED(hr)) hr = m_pVisual->SetContent(m_pD2DBitmap);
// 10. Добавляем визуал в корень дерева
if (SUCCEEDED(hr))
hr = m_pDCompTarget->SetRoot(m_pVisual);
// 11. Создаём анимации if (SUCCEEDED(hr)) hr = CreateAnimations();
// 12. Применяем анимации к визуалу
if (SUCCEEDED(hr))
{
// Анимация положения по X: перемещаемся от 0 до 500
hr = m_pVisual->SetOffsetX(m_pAnimX);
}
if (SUCCEEDED(hr))
{
// Анимация прозрачности: применяем эффект с анимацией
CComPtr<IDCompositionEffect> pEffect;
hr = m_pDCompDevice->CreateEffect(&pEffect);
if (SUCCEEDED(hr))
{
// Устанавливаем анимацию для свойства Opacity эффекта
hr = pEffect->SetOpacity(m_pAnimOpacity);
if (SUCCEEDED(hr))
hr = m_pVisual->SetEffect(pEffect);
}
}
// 13. Фиксируем все изменения — анимация запускается![reference:4]
if (SUCCEEDED(hr))
hr = m_pDCompDevice->Commit();
return hr;
}Создание контента (квадрата). Функция, выполняющая отрисовку квадрата в битмапе Direct2D:
HRESULT CreateBitmapContent() { HRESULT hr = S_OK;
// Размер битмапа D2D1_SIZE_U size = D2D1::SizeU(200, 200);
// Свойства битмапа
D2D1_BITMAP_PROPERTIES1 props = D2D1::BitmapProperties1(
D2D1_BITMAP_OPTIONS_TARGET, // Можно использовать как цель
D2D1::PixelFormat(DXGI_FORMAT_B8G8R8A8_UNORM, // Формат пикселей
D2D1_ALPHA_MODE_PREMULTIPLIED),
96.0f, 96.0f // DPI
);
// Создаём битмап в контексте D2D
hr = m_pD2DContext->CreateBitmap(size, nullptr, 0, props, &m_pD2DBitmap);
// Рисуем на битмапе
if (SUCCEEDED(hr))
{
// Устанавливаем битмап как цель рисования
m_pD2DContext->SetTarget(m_pD2DBitmap);
m_pD2DContext->BeginDraw();
m_pD2DContext->Clear(D2D1::ColorF(D2D1::ColorF::CornflowerBlue));
// Создаём кисть и рисуем квадрат с обводкой
m_pD2DContext->CreateSolidColorBrush(
D2D1::ColorF(D2D1::ColorF::OrangeRed),
&m_pBrush
);
D2D1_RECT_F rect = D2D1::RectF(20.0f, 20.0f, 180.0f, 180.0f);
m_pD2DContext->FillRectangle(rect, m_pBrush);
m_pD2DContext->DrawRectangle(rect, m_pBrush, 5.0f);
hr = m_pD2DContext->EndDraw(); m_pD2DContext->SetTarget(nullptr); }
return hr;
}Создание анимаций. Здесь мы определяем, как именно будут меняться свойства во времени. Вместо использования WAM мы создаём анимационные кривые вручную.
HRESULT CreateAnimations() { HRESULT hr = S_OK;
// --- Анимация движения по X (от 0 до 500 и обратно) ---
hr = m_pDCompDevice->CreateAnimation(&m_pAnimX);
if (SUCCEEDED(hr))
{
// Начинаем с 0
m_pAnimX->SetAbsoluteBeginTime(0);
m_pAnimX->SetKeyframes(nullptr, 0, nullptr);
// Первый сегмент: за 3 секунды перемещаемся к 500 с замедлением в конце
m_pAnimX->AddCubic(
0.0f, // Начальное смещение по времени
0.0f, // Начальное значение
3.0f, // Длительность (сек)
500.0f, // Конечное значение
0.0f, 0.0f, // Контрольные точки (0 = линейная)
1.0f, 1.0f
);
// Второй сегмент: за 3 секунды возвращаемся к 0
m_pAnimX->AddCubic(
0.0f,
500.0f,
3.0f,
0.0f,
0.0f, 0.0f,
1.0f, 1.0f
);
// Указываем, что анимация должна повторяться бесконечно
m_pAnimX->SetRepeatCount(-1);
}
// --- Анимация прозрачности (от 0.2 до 1.0) ---
hr = m_pDCompDevice->CreateAnimation(&m_pAnimOpacity);
if (SUCCEEDED(hr))
{
m_pAnimOpacity->SetAbsoluteBeginTime(0);
// Плавно увеличиваем прозрачность от 0.2 до 1.0 за 2 секунды
m_pAnimOpacity->AddCubic(
0.0f, 0.2f, 2.0f, 1.0f,
0.0f, 0.0f, 1.0f, 1.0f
);
// Плавно уменьшаем обратно до 0.2 за 2 секунды
m_pAnimOpacity->AddCubic(
0.0f, 1.0f, 2.0f, 0.2f,
0.0f, 0.0f, 1.0f, 1.0f
);
// Бесконечное повторение m_pAnimOpacity->SetRepeatCount(-1); }
return hr; }Пояснение аргументов методов:
CreateAnimation- единственный аргумент - выходной указатель, в который записывается адрес созданного объекта анимации (IDCompositionAnimation*).
SetAbsoluteBeginTime- устанавливает абсолютное время начала анимации относительно глобальной временной шкалы DirectComposition. Передача 0 означает, что анимация начинается с момента применения к свойству (то есть с момента вызова Commit).
SetKeyframes
Этот метод предназначен для задания ключевых кадров (более сложный способ построения анимации). В приведённом коде переданы значения nullptr, 0, nullptr - это фактически сбрасывает предыдущие ключевые кадры (если они были). Аргументы метода:
первый - массив элементов
второй - количество элементов
третий - отдельный ключевой кадр (в данном случае не используется).
AddCubic(добавление кубического сегмента)
Первый сегмент (движение по оси X от 0 до 500 за 3 секунды):
аргумент 1 - смещение внутри анимации, с которого начинается сегмент (0.0f);
аргумент 2 - значение, соответствующее началу сегмента (0.0f);
аргумент 3 - длительность сегмента в секундах (3.0f);
аргумент 4 - конечное значение сегмента (координата X станет равной 500.0f);
аргументы 5-6 - координаты первой контрольной точки кубической кривой Безье;
аргументы 7-8 - координаты второй контрольной точки.
Второй сегмент (движение от 500 обратно к 0 за 3 секунды):
beginValue = 500.0f - старт с 500;
endValue = 0.0f - возврат к 0;
длительность - снова 3 секунды;
контрольные точки такие же (0,0 и 1,1), что обеспечивает плавный старт и финиш.
SetRepeatCount
Задаёт количество повторений анимации после её первого выполнения. Значение -1 (передаётся как UINT, но в коде используется литерал -1, который интерпретируется как 0xFFFFFFFF) означает бесконечное повторение.
Анимация прозрачности (второй блок) создаётся аналогично - через m_pAnimOpacity.
Важно отметить: после вызова Commit() анимация начинает выполняться автоматически, без какого-либо участия приложения. Вам не нужно организовывать цикл обновления, вызывать BeginDraw/EndDraw в каждом кадре или синхронизироваться с WAM. DirectComposition обрабатывает анимацию в отдельном потоке на видеокарте.
Пока анимация работает, ваше приложение может выполнять любые другие задачи: обрабатывать пользовательский ввод, загружать данные или даже находиться в режиме ожидания - частота кадров при этом сохраняется постоянной.
На этом всё. По сути, вы можете создать свой аналог этих библиотек. Думаю, следующая статья будет посвящена эффектам и слоям, а потом будет создан движок игры, например, как в Stardew Valley, опираясь исключительно на свои же статьи, показав, что не так страшен чёрт, как его рисуют, и что даже такая низкоуровневая API, как Direct2D, вполне подходит инди-разработчикам.
Но, возможно, перед этим я рассмотрю XAudio2 (чтобы в игре был звук), а также сетевую часть (Winsock).
По сути, основная цель - создать игру на тех инструментах, которые рекомендует Microsoft.При желании материально поддержать перевод и структурирование информации - средства можете отправить через сбор в ЮМани.Теги:• C++
• Windows
• Direct 2D
• разработка игрХабы:• Windows
• C++
• Разработка игр
Получайте больше инсайтов о систематизации бизнеса
Подписывайтесь на Telegram-канал Business Operations — ежедневные материалы о бизнес-процессах, операционном управлении и повышении эффективности
💬 Подписаться на канал→ Оригинальная статья