Skip to content
  • Categories
  • Recent
  • Tags
  • Popular
  • Users
  • Groups
  • Search
  • Get Qt Extensions
  • Unsolved
Collapse
Brand Logo
  1. Home
  2. Qt Development
  3. General and Desktop
  4. qvideowidget causes mouse event exception

qvideowidget causes mouse event exception

Scheduled Pinned Locked Moved Unsolved General and Desktop
1 Posts 1 Posters 184 Views
  • Oldest to Newest
  • Newest to Oldest
  • Most Votes
Reply
  • Reply as topic
Log in to reply
This topic has been deleted. Only users with topic management privileges can see it.
  • O Offline
    O Offline
    onecodeWorld
    wrote on last edited by
    #1

    The following code is a borderless window class:

    
    EZFrameless::EZFrameless(QWidget *parent)
        : QWidget{parent}
        , d_ptr(new EZFramelessPrivate(this))
    {
        Q_D(EZFrameless);
        setWindowFlags(windowFlags() | Qt::FramelessWindowHint | Qt::WindowSystemMenuHint);
    
        setResizeable(d->m_bResizeable);
        setMinimumSize(400, 500);
    
        connect(windowHandle(), &QWindow::screenChanged, this, [=] {
            auto hWnd = reinterpret_cast<HWND>(windowHandle()->winId());
            SetWindowPos(hWnd, NULL, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_FRAMECHANGED);
        });
    }
    
    void EZFrameless::setResizeable(bool resizeable)
    {
        Q_D(EZFrameless);
        d->m_bResizeable = resizeable;
        HWND  hwnd = (HWND) this->winId();
        DWORD style = ::GetWindowLong(hwnd, GWL_STYLE);
        if (d->m_bResizeable) {
            ::SetWindowLong(hwnd, GWL_STYLE, style | WS_THICKFRAME | WS_CAPTION | WS_MAXIMIZEBOX);
        } else {
            ::SetWindowLong(hwnd, GWL_STYLE, (style & ~WS_MAXIMIZEBOX) | WS_CAPTION);
        }
    
        const int     num = 4;
        const MARGINS shadow = {num, num, num, num};
        DwmExtendFrameIntoClientArea(HWND(winId()), &shadow);
    }
    
    void EZFrameless::setTitlebarWidget(EZTitleBarBase *widget)
    {
        Q_D(EZFrameless);
        d->m_titlebar = widget;
    }
    
    bool EZFrameless::nativeEvent(const QByteArray &eventType, void *message, long *result)
    {
        Q_D(EZFrameless);
        MSG *msg = reinterpret_cast<MSG *>(message);
        switch (msg->message) {
        // To manage the layout and size of the non-client area of a window (i.e., the window's borders, title bar, etc.).
        case WM_NCCALCSIZE: {
            const auto clientRect = msg->wParam == FALSE
                                        ? reinterpret_cast<LPRECT>(msg->lParam)
                                        : &(reinterpret_cast<LPNCCALCSIZE_PARAMS>(msg->lParam))->rgrc[0];
            d->m_widgetRect = QRect(0,
                                    0,
                                    clientRect->right - clientRect->left,
                                    clientRect->bottom - clientRect->top);
            // To resolve the issue of abnormal refresh of the right border when horizontally stretching the window.
            if (clientRect->top != 0)
                clientRect->top -= 1;
    
            if (::IsZoomed(msg->hwnd)) {
                // To solve the issue where the actual size of the window exceeds the screen size after maximizing.
                const int borderOutside = 8;
                clientRect->left += borderOutside;
                clientRect->right -= borderOutside;
                clientRect->top += borderOutside;
                clientRect->bottom -= borderOutside;
            }
            *result = WVR_REDRAW;
            return true;
        } break;
        case WM_NCHITTEST: {
            *result = 0;
            long x = GET_X_LPARAM(msg->lParam);
            long y = GET_Y_LPARAM(msg->lParam);
    
            POINT nativeLocalPos{x, y};
            ::ScreenToClient(msg->hwnd, &nativeLocalPos);
            QPoint pt(nativeLocalPos.x, nativeLocalPos.y);
    
            const int borderWidth = 5;
    
            if (d->m_bResizeable) {
                bool bResizeWidth = minimumWidth() != maximumWidth();
                bool bResizeHeight = minimumHeight() != maximumHeight();
    
                if (bResizeWidth) {
                    // left border
                    if (pt.x() >= d->m_widgetRect.left()
                        && pt.x() < d->m_widgetRect.left() + borderWidth)
                        *result = HTLEFT;
                    // right border
                    if (pt.x() < d->m_widgetRect.right()
                        && pt.x() >= d->m_widgetRect.right() - borderWidth)
                        *result = HTRIGHT;
                }
                if (bResizeHeight) {
                    // top border
                    if (pt.y() >= d->m_widgetRect.top() && pt.y() < d->m_widgetRect.top() + borderWidth)
                        *result = HTTOP;
                    // bottom border
                    if (pt.y() < d->m_widgetRect.bottom()
                        && pt.y() >= d->m_widgetRect.bottom() - borderWidth)
                        *result = HTBOTTOM;
                }
                if (bResizeWidth && bResizeHeight) {
                    // left top corner
                    if (pt.x() >= d->m_widgetRect.left()
                        && pt.x() < d->m_widgetRect.left() + borderWidth
                        && pt.y() >= d->m_widgetRect.top()
                        && pt.y() < d->m_widgetRect.top() + borderWidth) {
                        *result = HTTOPLEFT;
                    }
                    // right top corner
                    if (pt.x() < d->m_widgetRect.right()
                        && pt.x() >= d->m_widgetRect.right() - borderWidth
                        && pt.y() >= d->m_widgetRect.top()
                        && pt.y() < d->m_widgetRect.top() + borderWidth) {
                        *result = HTTOPRIGHT;
                    }
                    // right bottom corner
                    if (pt.x() < d->m_widgetRect.right()
                        && pt.x() >= d->m_widgetRect.right() - borderWidth
                        && pt.y() < d->m_widgetRect.bottom()
                        && pt.y() >= d->m_widgetRect.bottom() - borderWidth) {
                        *result = HTBOTTOMRIGHT;
                    }
                    // left bottom corner
                    if (pt.x() >= d->m_widgetRect.left()
                        && pt.x() < d->m_widgetRect.left() + borderWidth
                        && pt.y() < d->m_widgetRect.bottom()
                        && pt.y() >= d->m_widgetRect.bottom() - borderWidth) {
                        *result = HTBOTTOMLEFT;
                    }
                }
            }
            if (0 != *result)
                return true;
    
            if (d->m_titlebar->isMovableArea(pt)) {
                *result = HTCAPTION;
                return true;
            }
        } break;
        default:
            break;
        }
        return QWidget::nativeEvent(eventType, message, result);
    }
    
    

    class Widget inherits EZFrameless, vertically layouts the custom title bar Titlbar (inherits EZTitleBarBase) and QVideoWidget objects to implement a simple player interface.
    There is no exception when running the program at this time.
    Until the following code is called, the EZFrameless::nativeEvent cannot effectively capture the WM_NCHITTEST message, resulting in the mouse being unable to drag the window.

    QMediaPlayer m_player = new QMediaPlayer(this);
    m_player->setVideoOutput(ui->videoWidget);
    

    But as long as the vertical size of the window is changed after the window is displayed, it will return to normal.
    Another problem is that when the window is zoomed, other controls in the window will appear ghosted.
    Who knows why the setVideoOutput function causes a series of exceptions after setting the output window?
    If you want to make a player, should you abandon Qt MediaPlayer and use a third-party library?

    1 Reply Last reply
    0

    • Login

    • Login or register to search.
    • First post
      Last post
    0
    • Categories
    • Recent
    • Tags
    • Popular
    • Users
    • Groups
    • Search
    • Get Qt Extensions
    • Unsolved