使用DirectDraw显示YUV视频

2024-01-13 01:58

本文主要是介绍使用DirectDraw显示YUV视频,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文出处: http://hi.baidu.com/sdlyfdy/blog/item/06e3c70cc0e57cec36d12297.html/cmtid/fa88294c57129204b2de0552

#ifndef _DDRAWDISP_H
#define _DDRAWDISP_H

#include <ddraw.h>

class CDDrawDisplay
{
public:
BOOL DDrawDispInit(HWND hWnd, int src_width, int src_height, int dst_width, int dst_height);
void DDrawDispReinit(HWND hWnd, int src_width, int src_height, int dst_width, int dst_height);
void DDrawDispDone();
void DDrawUpdateDisp(unsigned char *src[3], int stride[3]);

~CDDrawDisplay();

protected:

protected:
int m_src_width;
int m_src_height;
int m_dst_top;
int m_dst_left;
int m_dst_width;
int m_dst_height;

HWND m_hWnd;
LPDIRECTDRAW            m_lpDD;   // DirectDraw 对象指针
LPDIRECTDRAWSURFACE     m_lpDDSPrimary; // DirectDraw 主表面指针
LPDIRECTDRAWSURFACE     m_lpDDSOverlay; // DirectDraw 离屏表面指针

};

#endif


#include <windows.h>
#include "DDrawDisp.h"

#pragma comment(lib,"ddraw.lib")

static DDPIXELFORMAT ddpfOverlayFormats[] = {
    {sizeof(DDPIXELFORMAT), DDPF_FOURCC, MAKEFOURCC('Y','V','1','2'),0,0,0,0,0}, // YV12
    {sizeof(DDPIXELFORMAT), DDPF_FOURCC, MAKEFOURCC('U','Y','V','Y'),0,0,0,0,0}, // UYVY
    //{sizeof(DDPIXELFORMAT), DDPF_RGB, 0, 16, 0x7C00, 0x03e0, 0x001F, 0},  
     // 16-bit RGB 5:5:5
    //{sizeof(DDPIXELFORMAT), DDPF_RGB, 0, 16, 0xF800, 0x07e0, 0x001F, 0}   
     // 16-bit RGB 5:6:5
};

#define PF_TABLE_SIZE (sizeof(ddpfOverlayFormats) / sizeof(ddpfOverlayFormats[
0]))


BOOL CDDrawDisplay::DDrawDispInit(HWND hWnd, int src_width, int src_height,

     int dst_width, int dst_height)
{
RECT rect;

GetClientRect(hWnd, &rect);
// rect.right = GetSystemMetrics(SM_CXSCREEN);
// rect.bottom = GetSystemMetrics(SM_CYSCREEN);

m_src_width = src_width;
m_src_height = src_height;
m_hWnd = hWnd;

if (dst_width > rect.right)
dst_width = rect.right;
if (dst_height > rect.bottom)
dst_height = rect.bottom;

if (dst_width == -1) // Full Screen
{
if (rect.right*src_height/src_width > rect.bottom)
{
   dst_height = rect.bottom;
   dst_width = rect.bottom * src_width / src_height;
}
else
{
   dst_width = rect.right;
   dst_height = rect.right * src_height / src_width;
}
}

m_dst_left = (rect.right-dst_width)/2;
m_dst_top = (rect.bottom-dst_height)/2;
m_dst_width = dst_width;
m_dst_height = dst_height;

// 创建DirectCraw对象
if ( DirectDrawCreate(NULL, &m_lpDD, NULL) != DD_OK )
return FALSE;

// 设置协作层
if ( m_lpDD->SetCooperativeLevel(hWnd, DDSCL_NORMAL) != DD_OK )
return FALSE;

DDSURFACEDESC   ddsd;   // DirectDraw 表面描述

// 创建主表面
ZeroMemory(&ddsd, sizeof(ddsd));
ddsd.dwSize = sizeof(ddsd);
ddsd.dwFlags = DDSD_CAPS ;
ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;

if ( m_lpDD->CreateSurface(&ddsd, &m_lpDDSPrimary, NULL) != DD_OK )
return FALSE;

// 创建离屏表面对象
ZeroMemory(&ddsd, sizeof(ddsd));
    ddsd.dwSize = sizeof(ddsd);
    ddsd.ddsCaps.dwCaps = DDSCAPS_VIDEOMEMORY | DDSCAPS_OVERLAY;
    ddsd.dwFlags = DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH | DDSD_PIXELFORMAT;
    ddsd.dwWidth = src_width;
    ddsd.dwHeight = src_height;

    int i = 0;
    HRESULT                     hRet;
    do {
        ddsd.ddpfPixelFormat = ddpfOverlayFormats[i];
        hRet = m_lpDD->CreateSurface(&ddsd, &m_lpDDSOverlay, NULL);
    } while (hRet != DD_OK && (++i < PF_TABLE_SIZE));

    if (hRet != DD_OK)
        return FALSE;

return TRUE;
}

void CDDrawDisplay::DDrawDispReinit(HWND hWnd, int src_width, int src_height,

       int dst_width, int dst_height)
{
DDrawDispDone();
DDrawDispInit(hWnd, src_width, src_height, dst_width, dst_height);
}

void CDDrawDisplay::DDrawDispDone()
{
    if (m_lpDDSOverlay != NULL)
    {
        // Use UpdateOverlay() with the DDOVER_HIDE flag to remove an overlay

        // from the display.
        m_lpDDSOverlay->UpdateOverlay(NULL, m_lpDDSPrimary, NULL, DDOVER_HIDE,
NULL);
        m_lpDDSOverlay->Release();
        m_lpDDSOverlay = NULL;
    }

    if (m_lpDDSPrimary != NULL)
    {
        m_lpDDSPrimary->Release();
        m_lpDDSPrimary = NULL;
    }

    if (m_lpDD != NULL)
    {
        m_lpDD->Release();
        m_lpDD = NULL;
    }
}

CDDrawDisplay::~CDDrawDisplay()
{
DDrawDispDone();
}

void CDDrawDisplay::DDrawUpdateDisp(unsigned char *src[3], int stride[3])
{
HRESULT     hRet;   // DirectDraw 函数返回值
DDSURFACEDESC   ddsd;   // DirectDraw 表面描述

    memset(&ddsd, 0, sizeof(ddsd));
    ddsd.dwSize = sizeof(ddsd);
   
// Lock down the surface so we can modify it's contents.
    hRet = m_lpDDSOverlay->Lock( NULL, &ddsd, DDLOCK_WAITNOTBUSY, NULL);
    if (FAILED(hRet))
return ;

LPBYTE lpSurf = (LPBYTE)ddsd.lpSurface;

if(lpSurf) {
int i;

// fill Y data
for(i = 0; i < m_src_height; i++)
{
   memcpy(lpSurf, src[0], m_src_width);
   src[0] += stride[0];
   lpSurf += ddsd.lPitch;
}

// fill V data
for(i = 0; i < m_src_height / 2; i++)
{
   memcpy(lpSurf, src[2], m_src_width / 2);
   src[2] += stride[2];
   lpSurf += ddsd.lPitch / 2;
}

// fill U data
for(i = 0; i < m_src_height / 2; i++)
{
   memcpy(lpSurf, src[1], m_src_width / 2);
   src[1] += stride[1];
   lpSurf += ddsd.lPitch / 2;
}
}

m_lpDDSOverlay->Unlock(NULL);

RECT rs, rd;
rs.left = rs.top = 0;
rs.right = m_src_width;
rs.bottom = m_src_height;

rd.left = m_dst_left;
rd.top = m_dst_top;
rd.right = m_dst_left + m_dst_width;
rd.bottom = m_dst_top + m_dst_height;

hRet = m_lpDDSOverlay->UpdateOverlay(&rs, m_lpDDSPrimary, &rd, DDOVER_SHOW, NULL);
}
 

 

这篇关于使用DirectDraw显示YUV视频的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/599877

相关文章

Java使用SLF4J记录不同级别日志的示例详解

《Java使用SLF4J记录不同级别日志的示例详解》SLF4J是一个简单的日志门面,它允许在运行时选择不同的日志实现,这篇文章主要为大家详细介绍了如何使用SLF4J记录不同级别日志,感兴趣的可以了解下... 目录一、SLF4J简介二、添加依赖三、配置Logback四、记录不同级别的日志五、总结一、SLF4J

使用Python实现一个优雅的异步定时器

《使用Python实现一个优雅的异步定时器》在Python中实现定时器功能是一个常见需求,尤其是在需要周期性执行任务的场景下,本文给大家介绍了基于asyncio和threading模块,可扩展的异步定... 目录需求背景代码1. 单例事件循环的实现2. 事件循环的运行与关闭3. 定时器核心逻辑4. 启动与停

如何使用Nginx配置将80端口重定向到443端口

《如何使用Nginx配置将80端口重定向到443端口》这篇文章主要为大家详细介绍了如何将Nginx配置为将HTTP(80端口)请求重定向到HTTPS(443端口),文中的示例代码讲解详细,有需要的小伙... 目录1. 创建或编辑Nginx配置文件2. 配置HTTP重定向到HTTPS3. 配置HTTPS服务器

Java使用ANTLR4对Lua脚本语法校验详解

《Java使用ANTLR4对Lua脚本语法校验详解》ANTLR是一个强大的解析器生成器,用于读取、处理、执行或翻译结构化文本或二进制文件,下面就跟随小编一起看看Java如何使用ANTLR4对Lua脚本... 目录什么是ANTLR?第一个例子ANTLR4 的工作流程Lua脚本语法校验准备一个Lua Gramm

Java Optional的使用技巧与最佳实践

《JavaOptional的使用技巧与最佳实践》在Java中,Optional是用于优雅处理null的容器类,其核心目标是显式提醒开发者处理空值场景,避免NullPointerExce... 目录一、Optional 的核心用途二、使用技巧与最佳实践三、常见误区与反模式四、替代方案与扩展五、总结在 Java

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

Qt中QUndoView控件的具体使用

《Qt中QUndoView控件的具体使用》QUndoView是Qt框架中用于可视化显示QUndoStack内容的控件,本文主要介绍了Qt中QUndoView控件的具体使用,具有一定的参考价值,感兴趣的... 目录引言一、QUndoView 的用途二、工作原理三、 如何与 QUnDOStack 配合使用四、自

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

使用Python构建一个Hexo博客发布工具

《使用Python构建一个Hexo博客发布工具》虽然Hexo的命令行工具非常强大,但对于日常的博客撰写和发布过程,我总觉得缺少一个直观的图形界面来简化操作,下面我们就来看看如何使用Python构建一个... 目录引言Hexo博客系统简介设计需求技术选择代码实现主框架界面设计核心功能实现1. 发布文章2. 加

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的