MFC音视频播放器-支持电子放大等功能

前言

本播放器在VS2019下开发,使用ffmpeg+D3D实现视频播放渲染功能。同时本播放器支持录像功能、截图功能、音视频播放功能、码流信息显示、电子放大功能等。D3D的渲染同时支持surface和texture两种方式,电子放大功能是在D3D Texture方式下进行实现。以下为该功能的一些核心代码,具体可以通过本文末的链接进行工程源码下载。

一、播放器电子放大功能操作

步骤一:播放RTSP 视频流后,点击开启电子放大功能按钮,如下图:
步骤二:点击按钮后,视频上会出现电子放大的变倍信息,本功能支持鼠标滚轮进行图形放大与缩小,也支持鼠标拖动视频进行显示,如下图:
步骤三:鼠标滚轮操作电子放大的放大缩小功能,如下图:
步骤四:按住鼠标左键后,可以拖动视频进行显示,如下图:
步骤五:点击关闭电子放大按钮后,恢复正常显示:

二、电子放大相关源码

CVideoPlayer鼠标事件:

复制代码
void CVideoPlayer::OnLButtonDown(UINT nFlags, CPoint point)
{
    myprint("OnLButtonDown");
    SetFocus();

    if (m_bDigitalZoom) // Digital Zoom
    {
        m_D3DRender.SetStartPt(point);
        m_bLBtnDown = true;
    }

    CStatic::OnLButtonDown(nFlags, point);
}


void CVideoPlayer::OnLButtonUp(UINT nFlags, CPoint point)
{
    m_bLBtnDown = false;
    ReleaseCapture();
}


BOOL CVideoPlayer::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)
{    
    myprint("OnMouseWheel");
    CRect rcClient;
    GetClientRect(&rcClient);
    CPoint ptTemp = pt;
    ::ScreenToClient(m_hWnd, &ptTemp);
    if (!PtInRect(&rcClient, ptTemp))
        return FALSE;
    m_ptWheel = pt;
    ::ScreenToClient(m_hWnd, &m_ptWheel);

    if (m_bDigitalZoom)
    {
        zDelta /= WHEEL_DELTA;

        myprint("OnMouseWheel zDelta:%d", zDelta);
        if (zDelta > 0)
        {
            for (short i = 0; i < zDelta; i++)
                m_D3DRender.ZoomOut();
        }
        else
        {
            for (short i = 0; i > zDelta; i--)
                m_D3DRender.ZoomIn();
        }
        return TRUE;
    }
    return FALSE;
}

void CVideoPlayer::OnMouseMove(UINT nFlags, CPoint point)
{
    if (m_bPlaying == false)
        return;

    SetFocus();

    if (m_bLBtnDown && m_bDigitalZoom)
    {
        CRect rc;
        GetClientRect(&rc);
        m_D3DRender.MoveToPtStart(point, rc);
    }
    
    CStatic::OnMouseMove(nFlags, point);
}

CD3DRender处理:

复制代码
//zoom
void CD3DRender::ZoomIn()
{
    m_Radius += m_RadiusFirst / DEFAULT_ZOOM_TIMES;
    myprint("ZoomIn m_RadiusFirst:%lf m_Radius:%lf\n", m_RadiusFirst, m_Radius);

    if (m_Radius >= DEFAULT_ZOOM_TIMES)
    {
        m_Radius = DEFAULT_ZOOM_TIMES;
    }

    D3DXVECTOR3 position(-x_curCam, y_curCam, sinf(m_Angle) * m_Radius);
    D3DXVECTOR3 target(-x_curCam, y_curCam, 0.0f);
    D3DXVECTOR3 up(0.0f, 1.0f, 0.0f);
    D3DXMATRIX V;
    D3DXMatrixLookAtLH(&V, &position, &target, &up);

    m_pDirect3DDevice->SetTransform(D3DTS_VIEW, &V);
}

void CD3DRender::ZoomOut()
{
    m_Radius -= m_RadiusFirst / DEFAULT_ZOOM_TIMES;
    myprint("ZoomOut m_RadiusFirst:%lf m_Radius:%lf\n", m_RadiusFirst, m_Radius);

    if (m_Radius <= 1.1f)
    {
        m_Radius = 1.1f;
    }

    D3DXVECTOR3 position(-x_curCam, y_curCam, sinf(m_Angle) * m_Radius);
    D3DXVECTOR3 target(-x_curCam, y_curCam, 0.0f);
    D3DXVECTOR3 up(0.0f, 1.0f, 0.0f);
    D3DXMATRIX V;
    D3DXMatrixLookAtLH(&V, &position, &target, &up);

    m_pDirect3DDevice->SetTransform(D3DTS_VIEW, &V);
}

void CD3DRender::ResetZoom()
{
    x_Cam = x_curCam = 0;
    y_Cam = y_curCam = 0;

    m_Radius = m_RadiusFirst;
    m_Angle = m_AngleFirst;

    D3DXVECTOR3 position(cosf(m_Angle) * m_Radius, 0.0f, sinf(m_Angle) * m_Radius);
    D3DXVECTOR3 target(0.0f, 0.0f, 0.0f);
    D3DXVECTOR3 up(0.0f, 1.0f, 0.0f);
    D3DXMATRIX V;
    D3DXMatrixLookAtLH(&V, &position, &target, &up);

    m_pDirect3DDevice->SetTransform(D3DTS_VIEW, &V);
}

HRESULT CD3DRender::AppendZoomTextInfo(TCHAR* sText, CRect rcText) 
{
    my_mutex_lock(m_pDrawMutex);

    memset(&m_DigZoomInfo, 0, sizeof(StreamInfo_t));

    _sntprintf_s(m_DigZoomInfo.sText, sizeof(m_DigZoomInfo.sText), sizeof(m_DigZoomInfo.sText) - 1, "%s", sText);
    m_DigZoomInfo.rcText.top = rcText.top;
    m_DigZoomInfo.rcText.left = rcText.left;
    m_DigZoomInfo.rcText.right = rcText.right;
    m_DigZoomInfo.rcText.bottom = rcText.bottom;

    my_mutex_unlock(m_pDrawMutex);
    return 0;
}

HRESULT CD3DRender::ClearZoomTextInfo()
{
    my_mutex_lock(m_pDrawMutex);

    memset(&m_DigZoomInfo, 0, sizeof(StreamInfo_t));

    my_mutex_unlock(m_pDrawMutex);
    return 0;
}

void CD3DRender::D3DDrawZoomText()
{
    my_mutex_lock(m_pDrawMutex);

    m_pD3DFont->DrawText(NULL, m_DigZoomInfo.sText, -1, &m_DigZoomInfo.rcText, DT_LEFT/*DT_CENTER | DT_VCENTER*/, D3DCOLOR_XRGB(220, 20, 60));

    my_mutex_unlock(m_pDrawMutex);
}

void CD3DRender::SetStartPt(::CPoint pt)
{
    pt_start = pt;
    x_Cam = x_curCam;
    y_Cam = y_curCam;
}


void CD3DRender::MoveToPtStart(::CPoint pt, CRect rc)
{
    float x = x_Cam + 40.0f * ((float)m_Radius / 20.0f) * (float)(pt.x - pt_start.x) / (float)(rc.right - rc.left);
    float y = y_Cam + 40.0f * ((float)m_Radius / 20.0f) * (float)(pt.y - pt_start.y) / (float)(rc.bottom - rc.top);
    x_curCam = x;
    y_curCam = y;

    myprint("position( %f, %f, %f)  x_Cam = %f, y_Cam = %f\n", x, y, sinf(m_Angle) * m_Radius, x_Cam, y_Cam);
    myprint("rc(%d, %d, %d, %d)", rc.left, rc.top, rc.right, rc.bottom);
    D3DXVECTOR3 position(-x, y, sinf(m_Angle) * m_Radius);
    D3DXVECTOR3 target(-x, y, 0.0f);
    D3DXVECTOR3 up(0.0f, 1.0f, 0.0f);
    D3DXMATRIX V;
    D3DXMatrixLookAtLH(&V, &position, &target, &up);

    m_pDirect3DDevice->SetTransform(D3DTS_VIEW, &V);
}

三、相关下载

链接1: 可执行exe下载

链接2: 播放器工程源码下载

相关推荐
hu_yuchen1 小时前
C++:Lambda表达式
开发语言·c++·算法
一只鱼^_1 小时前
牛客周赛 Round 91
数据结构·c++·算法·数学建模·面试·贪心算法·动态规划
2401_858286112 小时前
CC52.【C++ Cont】滑动窗口
开发语言·数据结构·c++·算法·leetcode·滑动窗口
OSwich4 小时前
【虚幻C++笔记】碰撞检测
c++·笔记·虚幻
玖剹4 小时前
矩阵区域和 --- 前缀和
数据结构·c++·算法·leetcode·矩阵·动态规划·1024程序员节
这个懒人4 小时前
使用c++实现一个简易的量子计算,并向外提供服务
c++·量子计算
freyazzr4 小时前
Leetcode刷题 | Day50_图论02_岛屿问题01_dfs两种方法+bfs一种方法
数据结构·c++·算法·leetcode·深度优先·图论·广度优先
悦悦子a啊4 小时前
C++之string
开发语言·数据结构·c++
我想进大厂4 小时前
图论---LCA(倍增法)
数据结构·c++·算法·图论
明月看潮生4 小时前
青少年编程与数学 02-018 C++数据结构与算法 16课题、贪心算法
c++·算法·青少年编程·贪心算法·编程与数学