From 1b10672179ba4f8a5c0e8ea0a3390e228e3a9c04 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Mon, 20 Jul 2015 17:57:46 -0400 Subject: [PATCH 1/9] Video: Embed video into chat --- qtox.pro | 6 ++- src/video/netcamview.cpp | 15 +++++++ src/video/netcamview.h | 5 +++ src/video/videosurface.cpp | 8 ++-- src/widget/form/chatform.cpp | 3 +- src/widget/form/genericchatform.cpp | 7 +++- src/widget/form/genericchatform.h | 2 + src/widget/tool/movablewidget.cpp | 65 +++++++++++++++++++++++++++++ src/widget/tool/movablewidget.h | 40 ++++++++++++++++++ 9 files changed, 143 insertions(+), 8 deletions(-) create mode 100644 src/widget/tool/movablewidget.cpp create mode 100644 src/widget/tool/movablewidget.h diff --git a/qtox.pro b/qtox.pro index f2548a76e..748dfb753 100644 --- a/qtox.pro +++ b/qtox.pro @@ -493,7 +493,8 @@ SOURCES += \ src/widget/genericchatitemwidget.cpp \ src/widget/friendlistlayout.cpp \ src/widget/genericchatitemlayout.cpp \ - src/widget/categorywidget.cpp + src/widget/categorywidget.cpp \ + src/widget/tool/movablewidget.cpp HEADERS += \ src/audio/audio.h \ @@ -536,4 +537,5 @@ HEADERS += \ src/widget/genericchatitemwidget.h \ src/widget/friendlistlayout.h \ src/widget/genericchatitemlayout.h \ - src/widget/categorywidget.h + src/widget/categorywidget.h \ + src/widget/tool/movablewidget.h diff --git a/src/video/netcamview.cpp b/src/video/netcamview.cpp index eb0cc54e7..2e1bc270c 100644 --- a/src/video/netcamview.cpp +++ b/src/video/netcamview.cpp @@ -22,6 +22,7 @@ #include "src/video/videosurface.h" #include #include +#include "src/widget/tool/movablewidget.h" NetCamView::NetCamView(QWidget* parent) : QWidget(parent) @@ -34,6 +35,10 @@ NetCamView::NetCamView(QWidget* parent) videoSurface = new VideoSurface(this); mainLayout->addWidget(videoSurface); + + selfFrame = new MovableWidget(this); + selfFrame->setStyleSheet("background-color: red;"); + selfFrame->show(); } void NetCamView::show(VideoSource *source, const QString &title) @@ -60,3 +65,13 @@ void NetCamView::setTitle(const QString &title) { setWindowTitle(title); } + +void NetCamView::resizeEvent(QResizeEvent* event) +{ + QWidget::resizeEvent(event); + + float ratio = 1.33f; + int frameHeight = height() / 3.0f; + selfFrame->resize(frameHeight * ratio, frameHeight); + selfFrame->move(6, height() - selfFrame->height() - 6); +} diff --git a/src/video/netcamview.h b/src/video/netcamview.h index a208c10a6..576a4c422 100644 --- a/src/video/netcamview.h +++ b/src/video/netcamview.h @@ -26,6 +26,7 @@ class QHBoxLayout; struct vpx_image; class VideoSurface; class VideoSource; +class QFrame; class NetCamView : public QWidget { @@ -40,9 +41,13 @@ public: void setSource(VideoSource* s); void setTitle(const QString& title); +protected: + void resizeEvent(QResizeEvent* event) final override; + private: QHBoxLayout* mainLayout; VideoSurface* videoSurface; + QWidget* selfFrame; }; #endif // NETCAMVIEW_H diff --git a/src/video/videosurface.cpp b/src/video/videosurface.cpp index 290539ef3..3649230e0 100644 --- a/src/video/videosurface.cpp +++ b/src/video/videosurface.cpp @@ -6,14 +6,14 @@ qTox is libre software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. + (at your option) any later version. qTox is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. - You should have received a copy of the GNU General Public License + You should have received a copy of the GNU General Public License along with qTox. If not, see . */ @@ -105,7 +105,7 @@ void VideoSurface::paintEvent(QPaintEvent*) } QPainter painter(this); - painter.fillRect(painter.viewport(), Qt::black); + //painter.fillRect(painter.viewport(), Qt::blue); if (lastFrame) { QSize frameSize = lastFrame->getSize(); diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index 42f3c710f..d0461e6ad 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -1062,12 +1062,13 @@ void ChatForm::SendMessageStr(QString msg) Widget::getInstance()->updateFriendActivity(f); } } - +#include void ChatForm::showNetcam() { if (!netcam) netcam = new NetCamView(); netcam->show(Core::getInstance()->getVideoSourceFromCall(callId), f->getDisplayedName()); + bodySplitter->insertWidget(0, netcam); } void ChatForm::hideNetcam() diff --git a/src/widget/form/genericchatform.cpp b/src/widget/form/genericchatform.cpp index 4b2a4f47e..53dc8e36d 100644 --- a/src/widget/form/genericchatform.cpp +++ b/src/widget/form/genericchatform.cpp @@ -25,6 +25,7 @@ #include #include #include +#include #include "src/persistence/smileypack.h" #include "src/widget/emoticonswidget.h" @@ -125,7 +126,11 @@ GenericChatForm::GenericChatForm(QWidget *parent) micButton->setStyleSheet(micButtonStylesheet); setLayout(mainLayout); - mainLayout->addWidget(chatWidget); + + bodySplitter = new QSplitter(Qt::Vertical, this); + bodySplitter->addWidget(chatWidget); + + mainLayout->addWidget(bodySplitter); mainLayout->addLayout(mainFootLayout); mainLayout->setMargin(0); diff --git a/src/widget/form/genericchatform.h b/src/widget/form/genericchatform.h index 573d6757a..b7da8b2b5 100644 --- a/src/widget/form/genericchatform.h +++ b/src/widget/form/genericchatform.h @@ -40,6 +40,7 @@ class ChatLog; class MaskablePixmapWidget; class Widget; class FlyoutOverlayWidget; +class QSplitter; namespace Ui { class MainWindow; @@ -119,6 +120,7 @@ protected: QDateTime historyBaselineDate = QDateTime::currentDateTime(); // used by HistoryKeeper to load messages from t to historyBaselineDate (excluded) bool audioInputFlag; bool audioOutputFlag; + QSplitter* bodySplitter; }; #endif // GENERICCHATFORM_H diff --git a/src/widget/tool/movablewidget.cpp b/src/widget/tool/movablewidget.cpp new file mode 100644 index 000000000..6a132a586 --- /dev/null +++ b/src/widget/tool/movablewidget.cpp @@ -0,0 +1,65 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + qTox is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with qTox. If not, see . +*/ + +#include "movablewidget.h" +#include + +MovableWidget::MovableWidget(QWidget *parent) + : QWidget(parent) +{ + +} + +void MovableWidget::mousePressEvent(QMouseEvent* event) +{ + if (event->buttons() & Qt::LeftButton) + { + moving = true; + lastPoint = event->globalPos(); + } +} + +void MovableWidget::mouseMoveEvent(QMouseEvent* event) +{ + if (moving) + { + QPoint moveTo = pos() - (lastPoint - event->globalPos()); + + if (moveTo.x() < 0) + moveTo.setX(0); + + if (moveTo.y() < 0) + moveTo.setY(0); + + if (moveTo.x() + width() > parentWidget()->width()) + moveTo.setX(parentWidget()->width() - width()); + + if (moveTo.y() + height() > parentWidget()->height()) + moveTo.setY(parentWidget()->height() - height()); + + move(moveTo); + lastPoint = event->globalPos(); + } +} + +void MovableWidget::mouseReleaseEvent(QMouseEvent* event) +{ + if (!(event->buttons() & Qt::LeftButton)) + moving = false; +} diff --git a/src/widget/tool/movablewidget.h b/src/widget/tool/movablewidget.h new file mode 100644 index 000000000..2b06a0ad2 --- /dev/null +++ b/src/widget/tool/movablewidget.h @@ -0,0 +1,40 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + qTox is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with qTox. If not, see . +*/ + +#ifndef MOVABLEWIDGET_H +#define MOVABLEWIDGET_H + +#include + +class MovableWidget : public QWidget +{ +public: + MovableWidget(QWidget* parent = 0); + +protected: + void mousePressEvent(QMouseEvent* event); + void mouseMoveEvent(QMouseEvent* event); + void mouseReleaseEvent(QMouseEvent* event); + +private: + bool moving = false; + QPoint lastPoint; +}; + +#endif // MOVABLEWIDGET_H From 4c493b85a0d60d05ab5c8bf18f0734db157b060e Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 21 Jul 2015 14:38:43 -0400 Subject: [PATCH 2/9] Video: View self video during video call --- src/video/camerasource.cpp | 2 + src/video/camerasource.h | 3 ++ src/video/netcamview.cpp | 31 ++++++++++--- src/video/netcamview.h | 7 ++- src/video/videosurface.cpp | 74 ++++++++++++++++++++++++++++++- src/video/videosurface.h | 3 ++ src/widget/tool/movablewidget.cpp | 71 ++++++++++++++++++++++++----- src/widget/tool/movablewidget.h | 7 ++- 8 files changed, 176 insertions(+), 22 deletions(-) diff --git a/src/video/camerasource.cpp b/src/video/camerasource.cpp index 3558124f8..fde26ccd4 100644 --- a/src/video/camerasource.cpp +++ b/src/video/camerasource.cpp @@ -286,6 +286,8 @@ bool CameraSource::openDevice() while (!streamFuture.isRunning()) QThread::yieldCurrentThread(); + emit deviceOpened(); + return true; } diff --git a/src/video/camerasource.h b/src/video/camerasource.h index 234a67a7e..a87b98489 100644 --- a/src/video/camerasource.h +++ b/src/video/camerasource.h @@ -59,6 +59,9 @@ public: virtual bool subscribe() override; virtual void unsubscribe() override; +signals: + void deviceOpened(); + private: CameraSource(); ~CameraSource(); diff --git a/src/video/netcamview.cpp b/src/video/netcamview.cpp index 2e1bc270c..7b5018ae5 100644 --- a/src/video/netcamview.cpp +++ b/src/video/netcamview.cpp @@ -23,6 +23,7 @@ #include #include #include "src/widget/tool/movablewidget.h" +#include "camerasource.h" NetCamView::NetCamView(QWidget* parent) : QWidget(parent) @@ -36,15 +37,26 @@ NetCamView::NetCamView(QWidget* parent) mainLayout->addWidget(videoSurface); - selfFrame = new MovableWidget(this); + selfFrame = new MovableWidget(videoSurface); selfFrame->setStyleSheet("background-color: red;"); selfFrame->show(); + + selfVideoSurface = new VideoSurface(selfFrame); + selfVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); + selfVideoSurface->setMinimumSize(QSize(160, 120)); + selfVideoSurface->setSource(&CameraSource::getInstance()); + QHBoxLayout* camLayout = new QHBoxLayout(selfFrame); + camLayout->addWidget(selfVideoSurface); + camLayout->setMargin(0); + + connect(&CameraSource::getInstance(), &CameraSource::deviceOpened, this, &NetCamView::updateSize); } void NetCamView::show(VideoSource *source, const QString &title) { setSource(source); setTitle(title); + selfFrame->setBoundary(videoSurface->getRect()); QWidget::show(); } @@ -69,9 +81,16 @@ void NetCamView::setTitle(const QString &title) void NetCamView::resizeEvent(QResizeEvent* event) { QWidget::resizeEvent(event); - - float ratio = 1.33f; - int frameHeight = height() / 3.0f; - selfFrame->resize(frameHeight * ratio, frameHeight); - selfFrame->move(6, height() - selfFrame->height() - 6); + updateSize(); +} + +void NetCamView::updateSize() +{ + QSize frameSize = selfVideoSurface->getFrameSize(); + float ratio = frameSize.width() / static_cast(frameSize.height()); + QRect videoRect = videoSurface->getRect(); + int frameHeight = videoRect.height() / 3.0f; + selfFrame->resize(frameHeight * ratio, frameHeight); + selfFrame->setBoundary(videoRect); + selfVideoSurface->resize(selfFrame->size()); } diff --git a/src/video/netcamview.h b/src/video/netcamview.h index 576a4c422..593260406 100644 --- a/src/video/netcamview.h +++ b/src/video/netcamview.h @@ -27,6 +27,7 @@ struct vpx_image; class VideoSurface; class VideoSource; class QFrame; +class MovableWidget; class NetCamView : public QWidget { @@ -44,10 +45,14 @@ public: protected: void resizeEvent(QResizeEvent* event) final override; +private slots: + void updateSize(); + private: QHBoxLayout* mainLayout; VideoSurface* videoSurface; - QWidget* selfFrame; + VideoSurface* selfVideoSurface; + MovableWidget* selfFrame; }; #endif // NETCAMVIEW_H diff --git a/src/video/videosurface.cpp b/src/video/videosurface.cpp index 3649230e0..eda071e3e 100644 --- a/src/video/videosurface.cpp +++ b/src/video/videosurface.cpp @@ -28,7 +28,7 @@ VideoSurface::VideoSurface(QWidget* parent) , frameLock{false} , hasSubscribed{false} { - + //setMinimumWidth(160); } VideoSurface::VideoSurface(VideoSource *source, QWidget* parent) @@ -51,6 +51,49 @@ void VideoSurface::setSource(VideoSource *src) source = src; subscribe(); } +#include +QRect VideoSurface::getRect() +{ + // Fast lock + { + bool expected = false; + while (!frameLock.compare_exchange_weak(expected, true)) + expected = false; + } + + std::shared_ptr last = lastFrame; + frameLock = false; + + if (last) + { + QSize frameSize = lastFrame->getSize(); + QRect rect = this->rect(); + int width = frameSize.width()*rect.height()/frameSize.height(); + rect.setLeft((rect.width()-width)/2); + rect.setWidth(width); + return rect; + } + + return QRect(); +} + +QSize VideoSurface::getFrameSize() +{ + // Fast lock + { + bool expected = false; + while (!frameLock.compare_exchange_weak(expected, true)) + expected = false; + } + + QSize frameSize; + + if (lastFrame) + frameSize = lastFrame->getSize(); + + frameLock = false; + return frameSize; +} void VideoSurface::subscribe() { @@ -109,13 +152,40 @@ void VideoSurface::paintEvent(QPaintEvent*) if (lastFrame) { QSize frameSize = lastFrame->getSize(); - QRect rect = painter.viewport(); + QRect rect = this->rect(); int width = frameSize.width()*rect.height()/frameSize.height(); rect.setLeft((rect.width()-width)/2); rect.setWidth(width); QImage frame = lastFrame->toQImage(rect.size()); painter.drawImage(rect, frame, frame.rect(), Qt::NoFormatConversion); + //qDebug() << "VIDEO 2" << rect; } frameLock = false; } +#include +void VideoSurface::resizeEvent(QResizeEvent* event) +{ + QSize frameSize; + + // Fast lock + { + bool expected = false; + while (!frameLock.compare_exchange_weak(expected, true)) + expected = false; + } + + if (lastFrame) + { + frameSize = lastFrame->getSize(); + } + + frameLock = false; + + if (frameSize.isValid()) + { + float ratio = frameSize.height() / static_cast(frameSize.width()); + int width = ratio*event->size().width(); + setMaximumHeight(width); + } +} diff --git a/src/video/videosurface.h b/src/video/videosurface.h index 2b8fe17e7..3a5c26105 100644 --- a/src/video/videosurface.h +++ b/src/video/videosurface.h @@ -35,12 +35,15 @@ public: ~VideoSurface(); void setSource(VideoSource* src); //NULL is a valid option + QRect getRect(); + QSize getFrameSize(); protected: void subscribe(); void unsubscribe(); virtual void paintEvent(QPaintEvent * event) final override; + virtual void resizeEvent(QResizeEvent* event) final override; private slots: void onNewFrameAvailable(std::shared_ptr newFrame); diff --git a/src/widget/tool/movablewidget.cpp b/src/widget/tool/movablewidget.cpp index 6a132a586..943652e00 100644 --- a/src/widget/tool/movablewidget.cpp +++ b/src/widget/tool/movablewidget.cpp @@ -26,6 +26,15 @@ MovableWidget::MovableWidget(QWidget *parent) } +void MovableWidget::setBoundary(const QRect& boundary) +{ + boundaryRect = boundary; + + QPoint moveTo = pos(); + checkBoundary(moveTo); + move(moveTo); +} + void MovableWidget::mousePressEvent(QMouseEvent* event) { if (event->buttons() & Qt::LeftButton) @@ -40,18 +49,7 @@ void MovableWidget::mouseMoveEvent(QMouseEvent* event) if (moving) { QPoint moveTo = pos() - (lastPoint - event->globalPos()); - - if (moveTo.x() < 0) - moveTo.setX(0); - - if (moveTo.y() < 0) - moveTo.setY(0); - - if (moveTo.x() + width() > parentWidget()->width()) - moveTo.setX(parentWidget()->width() - width()); - - if (moveTo.y() + height() > parentWidget()->height()) - moveTo.setY(parentWidget()->height() - height()); + checkBoundary(moveTo); move(moveTo); lastPoint = event->globalPos(); @@ -63,3 +61,52 @@ void MovableWidget::mouseReleaseEvent(QMouseEvent* event) if (!(event->buttons() & Qt::LeftButton)) moving = false; } +#include +void MovableWidget::mouseDoubleClickEvent(QMouseEvent* event) +{ + if (!(event->buttons() & Qt::LeftButton)) + return; + + if (!graphicsEffect()) + { + QGraphicsOpacityEffect* opacityEffect = new QGraphicsOpacityEffect(this); + opacityEffect->setOpacity(0.5); + setGraphicsEffect(opacityEffect); + } + else + { + setGraphicsEffect(nullptr); + } +} + +void MovableWidget::checkBoundary(QPoint& point) const +{ + int x1, y1, x2, y2; + boundaryRect.getCoords(&x1, &y1, &x2, &y2); + + // Video boundary. + if (point.x() + width() < x1) + point.setX(x1 - width()); + + if (point.y() + height() x2) + point.setX(x2); + + if (point.y() > y2) + point.setY(y2); + + // Parent boundary. + if (point.x() < 0) + point.setX(0); + + if (point.y() < 0) + point.setY(0); + + if (point.x() + width() > parentWidget()->width()) + point.setX(parentWidget()->width() - width()); + + if (point.y() + height() > parentWidget()->height()) + point.setY(parentWidget()->height() - height()); +} diff --git a/src/widget/tool/movablewidget.h b/src/widget/tool/movablewidget.h index 2b06a0ad2..2821cab0f 100644 --- a/src/widget/tool/movablewidget.h +++ b/src/widget/tool/movablewidget.h @@ -25,16 +25,21 @@ class MovableWidget : public QWidget { public: - MovableWidget(QWidget* parent = 0); + MovableWidget(QWidget* parent); + void setBoundary(const QRect& boundary); protected: void mousePressEvent(QMouseEvent* event); void mouseMoveEvent(QMouseEvent* event); void mouseReleaseEvent(QMouseEvent* event); + void mouseDoubleClickEvent(QMouseEvent* event); private: + void checkBoundary(QPoint& point) const; + bool moving = false; QPoint lastPoint; + QRect boundaryRect; }; #endif // MOVABLEWIDGET_H From cd90d8d34b1f341ae2657639abb7bb2539d5f051 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Wed, 22 Jul 2015 14:50:39 -0400 Subject: [PATCH 3/9] Video: Add collapse/expand button, expand all the way, better self video output location --- src/video/netcamview.cpp | 108 +++++++++++++++++++++++----- src/video/netcamview.h | 10 +++ src/video/videosurface.cpp | 14 +++- src/video/videosurface.h | 9 ++- src/widget/form/chatform.cpp | 38 +++++++++- src/widget/form/chatform.h | 3 + src/widget/form/genericchatform.cpp | 9 ++- src/widget/form/genericchatform.h | 1 + src/widget/tool/movablewidget.cpp | 43 ++++++++--- src/widget/tool/movablewidget.h | 6 +- 10 files changed, 207 insertions(+), 34 deletions(-) diff --git a/src/video/netcamview.cpp b/src/video/netcamview.cpp index 7b5018ae5..5af313e29 100644 --- a/src/video/netcamview.cpp +++ b/src/video/netcamview.cpp @@ -18,53 +18,69 @@ */ #include "netcamview.h" +#include "camerasource.h" #include "src/core/core.h" #include "src/video/videosurface.h" -#include -#include #include "src/widget/tool/movablewidget.h" -#include "camerasource.h" +#include +#include +#include NetCamView::NetCamView(QWidget* parent) : QWidget(parent) , mainLayout(new QHBoxLayout()) + , selfFrame{nullptr} { - setLayout(mainLayout); + QVBoxLayout* layout = new QVBoxLayout(this); setWindowTitle(tr("Tox video")); setMinimumSize(320,240); videoSurface = new VideoSurface(this); + //mainLayout->addStretch(); mainLayout->addWidget(videoSurface); + //mainLayout->addStretch(); - selfFrame = new MovableWidget(videoSurface); - selfFrame->setStyleSheet("background-color: red;"); - selfFrame->show(); - - selfVideoSurface = new VideoSurface(selfFrame); + selfVideoSurface = new VideoSurface(this); selfVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); selfVideoSurface->setMinimumSize(QSize(160, 120)); selfVideoSurface->setSource(&CameraSource::getInstance()); - QHBoxLayout* camLayout = new QHBoxLayout(selfFrame); - camLayout->addWidget(selfVideoSurface); - camLayout->setMargin(0); - connect(&CameraSource::getInstance(), &CameraSource::deviceOpened, this, &NetCamView::updateSize); + connect(&CameraSource::getInstance(), &CameraSource::deviceOpened, [this]() + { + connect(selfVideoSurface, &VideoSurface::drewNewFrame, this, &NetCamView::updateSize); + }); + + QHBoxLayout* buttonLayout = new QHBoxLayout(); + buttonLayout->addStretch(); + button = new QPushButton(); + buttonLayout->addWidget(button); + connect(button, &QPushButton::clicked, this, &NetCamView::showMessageClicked); + + layout->addLayout(mainLayout); + layout->addLayout(buttonLayout); + + setShowMessages(false); } void NetCamView::show(VideoSource *source, const QString &title) { setSource(source); setTitle(title); - selfFrame->setBoundary(videoSurface->getRect()); QWidget::show(); + updateSize(); } void NetCamView::hide() { setSource(nullptr); + if (selfFrame) + selfFrame->deleteLater(); + + selfFrame = nullptr; + QWidget::hide(); } @@ -78,6 +94,22 @@ void NetCamView::setTitle(const QString &title) setWindowTitle(title); } +void NetCamView::setShowMessages(bool show, bool notify) +{ + if (show) + { + button->setText(tr("Show Messages")); + + if (notify) + button->setIcon(QIcon("://ui/chatArea/info.svg")); + } + else + { + button->setText(tr("Hide Messages")); + button->setIcon(QIcon()); + } +} + void NetCamView::resizeEvent(QResizeEvent* event) { QWidget::resizeEvent(event); @@ -85,12 +117,54 @@ void NetCamView::resizeEvent(QResizeEvent* event) } void NetCamView::updateSize() +{ + // Check there is room for a second video. + // If so, then we will show the user video there too. + //qDebug() << selfVideoSurface->getRect().size() == ; + bool hasRoom = selfVideoSurface->getRect().width() != 0 && videoSurface->getRect().width() * 2 < layout()->contentsRect().width() - layout()->margin(); + + if (mainLayout->indexOf(selfVideoSurface) != -1) + { + + if (!hasRoom) + { + selfFrame = new MovableWidget(videoSurface); + selfFrame->show(); + + QHBoxLayout* camLayout = new QHBoxLayout(selfFrame); + camLayout->addWidget(selfVideoSurface); + camLayout->setMargin(0); + + //selfFrame->setBoundary(videoSurface->getRect()); + updateFrameSize(); + } + } + else + { + if (hasRoom) + { + if (selfFrame) + selfFrame->deleteLater(); + + selfFrame = nullptr; + + mainLayout->addWidget(selfVideoSurface); + } + else if (selfFrame) + { + updateFrameSize(); + } + } + + disconnect(selfVideoSurface, &VideoSurface::drewNewFrame, this, &NetCamView::updateSize); +} + +void NetCamView::updateFrameSize() { QSize frameSize = selfVideoSurface->getFrameSize(); float ratio = frameSize.width() / static_cast(frameSize.height()); QRect videoRect = videoSurface->getRect(); int frameHeight = videoRect.height() / 3.0f; - selfFrame->resize(frameHeight * ratio, frameHeight); - selfFrame->setBoundary(videoRect); - selfVideoSurface->resize(selfFrame->size()); + //selfFrame->resize(frameHeight * ratio, frameHeight); + selfFrame->setBoundary(videoRect, QSize(frameHeight * ratio, frameHeight)); } diff --git a/src/video/netcamview.h b/src/video/netcamview.h index 593260406..fdc2d2c51 100644 --- a/src/video/netcamview.h +++ b/src/video/netcamview.h @@ -28,6 +28,7 @@ class VideoSurface; class VideoSource; class QFrame; class MovableWidget; +class QPushButton; class NetCamView : public QWidget { @@ -42,6 +43,12 @@ public: void setSource(VideoSource* s); void setTitle(const QString& title); +signals: + void showMessageClicked(); + +public slots: + void setShowMessages(bool show, bool notify = false); + protected: void resizeEvent(QResizeEvent* event) final override; @@ -49,10 +56,13 @@ private slots: void updateSize(); private: + void updateFrameSize(); + QHBoxLayout* mainLayout; VideoSurface* videoSurface; VideoSurface* selfVideoSurface; MovableWidget* selfFrame; + QPushButton* button; }; #endif // NETCAMVIEW_H diff --git a/src/video/videosurface.cpp b/src/video/videosurface.cpp index eda071e3e..6c5fb78a6 100644 --- a/src/video/videosurface.cpp +++ b/src/video/videosurface.cpp @@ -28,7 +28,7 @@ VideoSurface::VideoSurface(QWidget* parent) , frameLock{false} , hasSubscribed{false} { - //setMinimumWidth(160); + setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); } VideoSurface::VideoSurface(VideoSource *source, QWidget* parent) @@ -52,7 +52,7 @@ void VideoSurface::setSource(VideoSource *src) subscribe(); } #include -QRect VideoSurface::getRect() +QRect VideoSurface::getRect() const { // Fast lock { @@ -95,6 +95,11 @@ QSize VideoSurface::getFrameSize() return frameSize; } +QSize VideoSurface::sizeHint() const +{ + return getRect().size(); +} + void VideoSurface::subscribe() { if (source && !hasSubscribed) @@ -136,6 +141,8 @@ void VideoSurface::onNewFrameAvailable(std::shared_ptr newFrame) lastFrame = newFrame; frameLock = false; update(); + + emit drewNewFrame(); } void VideoSurface::paintEvent(QPaintEvent*) @@ -148,7 +155,7 @@ void VideoSurface::paintEvent(QPaintEvent*) } QPainter painter(this); - //painter.fillRect(painter.viewport(), Qt::blue); + //painter.fillRect(painter.viewport(), Qt::black); if (lastFrame) { QSize frameSize = lastFrame->getSize(); @@ -166,6 +173,7 @@ void VideoSurface::paintEvent(QPaintEvent*) #include void VideoSurface::resizeEvent(QResizeEvent* event) { + // Locks aspect ratio. QSize frameSize; // Fast lock diff --git a/src/video/videosurface.h b/src/video/videosurface.h index 3a5c26105..584c39336 100644 --- a/src/video/videosurface.h +++ b/src/video/videosurface.h @@ -35,9 +35,14 @@ public: ~VideoSurface(); void setSource(VideoSource* src); //NULL is a valid option - QRect getRect(); + QRect getRect() const; QSize getFrameSize(); + virtual QSize sizeHint() const override; + +signals: + void drewNewFrame(); + protected: void subscribe(); void unsubscribe(); @@ -51,7 +56,7 @@ private slots: private: VideoSource* source; std::shared_ptr lastFrame; - std::atomic_bool frameLock; ///< Fast lock for lastFrame + mutable std::atomic_bool frameLock; ///< Fast lock for lastFrame bool hasSubscribed; }; diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index d0461e6ad..d297d769c 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -31,6 +31,7 @@ #include #include #include +#include #include #include "chatform.h" #include "src/core/core.h" @@ -88,6 +89,10 @@ ChatForm::ChatForm(Friend* chatFriend) headTextLayout->addWidget(callDuration, 1, Qt::AlignCenter); callDuration->hide(); + chatWidget->setMinimumHeight(160); + connect(bodySplitter, &QSplitter::splitterMoved, this, &ChatForm::onSplitterMoved); + connect(this, &GenericChatForm::messageInserted, this, &ChatForm::onMessageInserted); + loadHistoryAction = menu.addAction(QString(), this, SLOT(onLoadHistory())); connect(Core::getInstance(), &Core::fileSendStarted, this, &ChatForm::startFileSend); @@ -931,6 +936,31 @@ void ChatForm::onLoadHistory() } } +void ChatForm::onSplitterMoved(int, int) +{ + if (netcam) + netcam->setShowMessages(bodySplitter->sizes()[1] == 0); +} + +void ChatForm::onMessageInserted() +{ + if (netcam && bodySplitter->sizes()[1] == 0) + netcam->setShowMessages(true, true); +} + +void ChatForm::onShowMessagesClicked() +{ + if (netcam) + { + if (bodySplitter->sizes()[1] == 0) + bodySplitter->setSizes({1, 1}); + else + bodySplitter->setSizes({1, 0}); + + onSplitterMoved(0, 0); + } +} + void ChatForm::startCounter() { if (!callDurationTimer) @@ -1062,13 +1092,16 @@ void ChatForm::SendMessageStr(QString msg) Widget::getInstance()->updateFriendActivity(f); } } -#include + void ChatForm::showNetcam() { if (!netcam) netcam = new NetCamView(); + netcam->show(Core::getInstance()->getVideoSourceFromCall(callId), f->getDisplayedName()); + connect(netcam, &NetCamView::showMessageClicked, this, &ChatForm::onShowMessagesClicked); bodySplitter->insertWidget(0, netcam); + bodySplitter->setCollapsible(0, false); } void ChatForm::hideNetcam() @@ -1083,4 +1116,7 @@ void ChatForm::hideNetcam() void ChatForm::retranslateUi() { loadHistoryAction->setText(tr("Load chat history...")); + + if (netcam) + netcam->setShowMessages(chatWidget->isVisible()); } diff --git a/src/widget/form/chatform.h b/src/widget/form/chatform.h index 2aaf72f04..63e03702e 100644 --- a/src/widget/form/chatform.h +++ b/src/widget/form/chatform.h @@ -100,6 +100,9 @@ private slots: void onScreenshotClicked(); void onScreenshotTaken(const QPixmap &pixmap); void doScreenshot(); + void onSplitterMoved(int pos, int index); + void onMessageInserted(); + void onShowMessagesClicked(); private: void retranslateUi(); diff --git a/src/widget/form/genericchatform.cpp b/src/widget/form/genericchatform.cpp index 53dc8e36d..8db137390 100644 --- a/src/widget/form/genericchatform.cpp +++ b/src/widget/form/genericchatform.cpp @@ -128,10 +128,14 @@ GenericChatForm::GenericChatForm(QWidget *parent) setLayout(mainLayout); bodySplitter = new QSplitter(Qt::Vertical, this); - bodySplitter->addWidget(chatWidget); + + QWidget* contentWidget = new QWidget(this); + QVBoxLayout* contentLayout = new QVBoxLayout(contentWidget); + contentLayout->addWidget(chatWidget); + contentLayout->addLayout(mainFootLayout); + bodySplitter->addWidget(contentWidget); mainLayout->addWidget(bodySplitter); - mainLayout->addLayout(mainFootLayout); mainLayout->setMargin(0); footButtonsSmall->addWidget(emoteButton); @@ -464,6 +468,7 @@ QString GenericChatForm::resolveToxId(const ToxId &id) void GenericChatForm::insertChatMessage(ChatMessage::Ptr msg) { chatWidget->insertChatlineAtBottom(std::dynamic_pointer_cast(msg)); + emit messageInserted(); } void GenericChatForm::hideEvent(QHideEvent* event) diff --git a/src/widget/form/genericchatform.h b/src/widget/form/genericchatform.h index b7da8b2b5..7b835e687 100644 --- a/src/widget/form/genericchatform.h +++ b/src/widget/form/genericchatform.h @@ -71,6 +71,7 @@ signals: void sendMessage(uint32_t, QString); void sendAction(uint32_t, QString); void chatAreaCleared(); + void messageInserted(); public slots: void focusInput(); diff --git a/src/widget/tool/movablewidget.cpp b/src/widget/tool/movablewidget.cpp index 943652e00..93d8a74e6 100644 --- a/src/widget/tool/movablewidget.cpp +++ b/src/widget/tool/movablewidget.cpp @@ -19,18 +19,34 @@ #include "movablewidget.h" #include +#include +#include MovableWidget::MovableWidget(QWidget *parent) : QWidget(parent) { } - -void MovableWidget::setBoundary(const QRect& boundary) +#include +void MovableWidget::setBoundary(const QRect& boundary, QSize size) { + int widthEdge = boundaryRect.x() - width(); + checkBoundaryLeft(widthEdge); + int widthRange = abs(widthEdge) * 2 + boundaryRect.width() - width(); + float xPercent = static_cast(x() - widthEdge) / widthRange; + qDebug() << xPercent << x() - widthEdge << widthRange; + float yPercent = static_cast(y()) / (boundaryRect.height() - height()); + + if (size.isValid()) + resize(size); + boundaryRect = boundary; + widthEdge = boundaryRect.x() - width(); + checkBoundaryLeft(widthEdge); QPoint moveTo = pos(); + moveTo.setX((abs(widthEdge) * 2 + boundaryRect.width() - width()) * xPercent); + moveTo.setY((boundaryRect.height() - height()) * yPercent); checkBoundary(moveTo); move(moveTo); } @@ -61,7 +77,7 @@ void MovableWidget::mouseReleaseEvent(QMouseEvent* event) if (!(event->buttons() & Qt::LeftButton)) moving = false; } -#include + void MovableWidget::mouseDoubleClickEvent(QMouseEvent* event) { if (!(event->buttons() & Qt::LeftButton)) @@ -81,12 +97,17 @@ void MovableWidget::mouseDoubleClickEvent(QMouseEvent* event) void MovableWidget::checkBoundary(QPoint& point) const { + if (boundaryRect.isNull()) + return; + int x1, y1, x2, y2; boundaryRect.getCoords(&x1, &y1, &x2, &y2); + x1 = point.x(); + checkBoundaryLeft(x1); + point.setX(x1); + // Video boundary. - if (point.x() + width() < x1) - point.setX(x1 - width()); if (point.y() + height() parentWidget()->height()) point.setY(parentWidget()->height() - height()); } + +void MovableWidget::checkBoundaryLeft(int &x) const +{ + if (x < 0) + x = 0; + + if (x + width() < boundaryRect.x()) + x = boundaryRect.x() - width(); +} diff --git a/src/widget/tool/movablewidget.h b/src/widget/tool/movablewidget.h index 2821cab0f..4d6d4920d 100644 --- a/src/widget/tool/movablewidget.h +++ b/src/widget/tool/movablewidget.h @@ -26,7 +26,7 @@ class MovableWidget : public QWidget { public: MovableWidget(QWidget* parent); - void setBoundary(const QRect& boundary); + void setBoundary(const QRect& boundary, QSize size = QSize()); protected: void mousePressEvent(QMouseEvent* event); @@ -36,6 +36,10 @@ protected: private: void checkBoundary(QPoint& point) const; + void checkBoundaryTop(int &x) const; + void checkBoundaryBottom(int &x) const; + void checkBoundaryLeft(int &x) const; + void checkBoundaryRight(int &x) const; bool moving = false; QPoint lastPoint; From 9d12c295be33ed7bb89a40649d50bae260f65100 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 11 Aug 2015 14:01:49 -0400 Subject: [PATCH 4/9] Keep aspect ratio of inner video --- qtox.pro | 6 +- src/mainwindow.ui | 4 +- src/video/netcamview.cpp | 109 +++++++------- src/video/netcamview.h | 5 +- src/video/videosurface.cpp | 36 ++--- src/video/videosurface.h | 11 +- src/widget/form/chatform.cpp | 3 + src/widget/tool/aspectratiowidget.cpp | 131 +++++++++++++++++ src/widget/tool/aspectratiowidget.h | 66 +++++++++ src/widget/tool/movablewidget.cpp | 201 ++++++++++++++++++++++---- src/widget/tool/movablewidget.h | 25 +++- src/widget/widget.cpp | 1 + 12 files changed, 479 insertions(+), 119 deletions(-) create mode 100644 src/widget/tool/aspectratiowidget.cpp create mode 100644 src/widget/tool/aspectratiowidget.h diff --git a/qtox.pro b/qtox.pro index 748dfb753..28ae6d613 100644 --- a/qtox.pro +++ b/qtox.pro @@ -494,7 +494,8 @@ SOURCES += \ src/widget/friendlistlayout.cpp \ src/widget/genericchatitemlayout.cpp \ src/widget/categorywidget.cpp \ - src/widget/tool/movablewidget.cpp + src/widget/tool/movablewidget.cpp \ + src/widget/tool/aspectratiowidget.cpp HEADERS += \ src/audio/audio.h \ @@ -538,4 +539,5 @@ HEADERS += \ src/widget/friendlistlayout.h \ src/widget/genericchatitemlayout.h \ src/widget/categorywidget.h \ - src/widget/tool/movablewidget.h + src/widget/tool/movablewidget.h \ + src/widget/tool/aspectratiowidget.h diff --git a/src/mainwindow.ui b/src/mainwindow.ui index 9e73284de..da3e67da6 100644 --- a/src/mainwindow.ui +++ b/src/mainwindow.ui @@ -1085,7 +1085,7 @@ QSplitter:handle{ 0 0 284 - 353 + 341 @@ -1850,7 +1850,7 @@ QSplitter:handle{ 0 0 775 - 22 + 26 diff --git a/src/video/netcamview.cpp b/src/video/netcamview.cpp index 5af313e29..55fb6ec5b 100644 --- a/src/video/netcamview.cpp +++ b/src/video/netcamview.cpp @@ -33,23 +33,39 @@ NetCamView::NetCamView(QWidget* parent) { QVBoxLayout* layout = new QVBoxLayout(this); setWindowTitle(tr("Tox video")); - setMinimumSize(320,240); videoSurface = new VideoSurface(this); + videoSurface->setSizePolicy(QSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum)); + videoSurface->setStyleSheet("background-color: blue;"); + videoSurface->setMinimum(128); - //mainLayout->addStretch(); - mainLayout->addWidget(videoSurface); - //mainLayout->addStretch(); + mainLayout->addStretch(1); + //QWidget* wid = new QWidget(this); + //wid->setMinimumSize(600, 600); + QVBoxLayout* horLayout = new QVBoxLayout(); + horLayout->addStretch(1); + horLayout->addWidget(videoSurface); + horLayout->addStretch(1); + mainLayout->addLayout(horLayout); - selfVideoSurface = new VideoSurface(this); + selfFrame = new MovableWidget(videoSurface); + selfFrame->show(); + + selfVideoSurface = new VideoSurface(selfFrame); selfVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); - selfVideoSurface->setMinimumSize(QSize(160, 120)); - selfVideoSurface->setSource(&CameraSource::getInstance()); + selfVideoSurface->setMouseTracking(true); - connect(&CameraSource::getInstance(), &CameraSource::deviceOpened, [this]() + QHBoxLayout* frameLayout = new QHBoxLayout(selfFrame); + frameLayout->addWidget(selfVideoSurface); + frameLayout->setMargin(0); + + //mainLayout->addWidget(selfVideoSurface); + mainLayout->addStretch(1); + + /*connect(&CameraSource::getInstance(), &CameraSource::deviceOpened, [this]() { connect(selfVideoSurface, &VideoSurface::drewNewFrame, this, &NetCamView::updateSize); - }); + });*/ QHBoxLayout* buttonLayout = new QHBoxLayout(); buttonLayout->addStretch(); @@ -57,24 +73,29 @@ NetCamView::NetCamView(QWidget* parent) buttonLayout->addWidget(button); connect(button, &QPushButton::clicked, this, &NetCamView::showMessageClicked); - layout->addLayout(mainLayout); + layout->addLayout(mainLayout, 1); layout->addLayout(buttonLayout); setShowMessages(false); + + setStyleSheet("NetCamView { background-color: #c1c1c1; }"); } void NetCamView::show(VideoSource *source, const QString &title) { setSource(source); + //selfVideoSurface->setSource(&CameraSource::getInstance()); setTitle(title); QWidget::show(); - updateSize(); + //updateSize(); } void NetCamView::hide() { + qDebug() << "jd"; setSource(nullptr); + selfVideoSurface->setSource(nullptr); if (selfFrame) selfFrame->deleteLater(); @@ -109,6 +130,14 @@ void NetCamView::setShowMessages(bool show, bool notify) button->setIcon(QIcon()); } } +#include +#include "src/widget/style.h" +void NetCamView::paintEvent(QPaintEvent*) +{ + QPainter painter(this); + painter.setBrush(Style::getColor(Style::ThemeDark)); + painter.drawRect(rect()); +} void NetCamView::resizeEvent(QResizeEvent* event) { @@ -118,53 +147,23 @@ void NetCamView::resizeEvent(QResizeEvent* event) void NetCamView::updateSize() { - // Check there is room for a second video. - // If so, then we will show the user video there too. - //qDebug() << selfVideoSurface->getRect().size() == ; - bool hasRoom = selfVideoSurface->getRect().width() != 0 && videoSurface->getRect().width() * 2 < layout()->contentsRect().width() - layout()->margin(); + qDebug() << videoSurface->size(); + QSize usableSize = mainLayout->contentsRect().size(); + int possibleWidth = usableSize.height() * videoSurface->getRatio(); - if (mainLayout->indexOf(selfVideoSurface) != -1) - { + QSize initial = videoSurface->sizeHint(); - if (!hasRoom) - { - selfFrame = new MovableWidget(videoSurface); - selfFrame->show(); + if (!initial.isValid()) + return; - QHBoxLayout* camLayout = new QHBoxLayout(selfFrame); - camLayout->addWidget(selfVideoSurface); - camLayout->setMargin(0); + if (possibleWidth > usableSize.width()) + videoSurface->setSizeHint(usableSize.width(), usableSize.width() / videoSurface->getRatio()); + else + videoSurface->setSizeHint(usableSize.height() * videoSurface->getRatio(), usableSize.height()); - //selfFrame->setBoundary(videoSurface->getRect()); - updateFrameSize(); - } - } - else - { - if (hasRoom) - { - if (selfFrame) - selfFrame->deleteLater(); + videoSurface->updateGeometry(); - selfFrame = nullptr; - - mainLayout->addWidget(selfVideoSurface); - } - else if (selfFrame) - { - updateFrameSize(); - } - } - - disconnect(selfVideoSurface, &VideoSurface::drewNewFrame, this, &NetCamView::updateSize); -} - -void NetCamView::updateFrameSize() -{ - QSize frameSize = selfVideoSurface->getFrameSize(); - float ratio = frameSize.width() / static_cast(frameSize.height()); - QRect videoRect = videoSurface->getRect(); - int frameHeight = videoRect.height() / 3.0f; - //selfFrame->resize(frameHeight * ratio, frameHeight); - selfFrame->setBoundary(videoRect, QSize(frameHeight * ratio, frameHeight)); + QSize newSize = videoSurface->sizeHint(); + QSizeF initialSize = initial; + selfFrame->setBoundary(newSize, initial, newSize.width() / initialSize.width(), newSize.height() / initialSize.height()); } diff --git a/src/video/netcamview.h b/src/video/netcamview.h index fdc2d2c51..3e610cd1b 100644 --- a/src/video/netcamview.h +++ b/src/video/netcamview.h @@ -21,6 +21,7 @@ #define NETCAMVIEW_H #include +#include "src/widget/tool/aspectratiowidget.h" class QHBoxLayout; struct vpx_image; @@ -50,9 +51,10 @@ public slots: void setShowMessages(bool show, bool notify = false); protected: + void paintEvent(QPaintEvent* event); void resizeEvent(QResizeEvent* event) final override; -private slots: +public slots: void updateSize(); private: @@ -63,6 +65,7 @@ private: VideoSurface* selfVideoSurface; MovableWidget* selfFrame; QPushButton* button; + bool e = false; }; #endif // NETCAMVIEW_H diff --git a/src/video/videosurface.cpp b/src/video/videosurface.cpp index 6c5fb78a6..f4b308cdb 100644 --- a/src/video/videosurface.cpp +++ b/src/video/videosurface.cpp @@ -21,14 +21,16 @@ #include "src/video/videoframe.h" #include #include - +#include VideoSurface::VideoSurface(QWidget* parent) - : QWidget{parent} + : AspectRatioWidget{parent} , source{nullptr} , frameLock{false} , hasSubscribed{false} { - setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); + qDebug() << "ddd" << minimumSize(); + setMinimum(128); + setSizeHint(128, 128); } VideoSurface::VideoSurface(VideoSource *source, QWidget* parent) @@ -51,7 +53,7 @@ void VideoSurface::setSource(VideoSource *src) source = src; subscribe(); } -#include +/*#include QRect VideoSurface::getRect() const { // Fast lock @@ -77,7 +79,7 @@ QRect VideoSurface::getRect() const return QRect(); } -QSize VideoSurface::getFrameSize() +QSize VideoSurface::getFrameSize() const { // Fast lock { @@ -93,12 +95,7 @@ QSize VideoSurface::getFrameSize() frameLock = false; return frameSize; -} - -QSize VideoSurface::sizeHint() const -{ - return getRect().size(); -} +}*/ void VideoSurface::subscribe() { @@ -140,6 +137,9 @@ void VideoSurface::onNewFrameAvailable(std::shared_ptr newFrame) lastFrame = newFrame; frameLock = false; + setRatio(lastFrame->getSize().width() / static_cast(lastFrame->getSize().height())); + ///setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding); + updateGeometry(); update(); emit drewNewFrame(); @@ -155,22 +155,22 @@ void VideoSurface::paintEvent(QPaintEvent*) } QPainter painter(this); - //painter.fillRect(painter.viewport(), Qt::black); + painter.fillRect(painter.viewport(), QColor(193, 193, 193)); if (lastFrame) { - QSize frameSize = lastFrame->getSize(); + /*QSize frameSize = lastFrame->getSize(); QRect rect = this->rect(); int width = frameSize.width()*rect.height()/frameSize.height(); rect.setLeft((rect.width()-width)/2); - rect.setWidth(width); + rect.setWidth(width);*/ - QImage frame = lastFrame->toQImage(rect.size()); - painter.drawImage(rect, frame, frame.rect(), Qt::NoFormatConversion); + QImage frame = lastFrame->toQImage(rect().size()); + painter.drawImage(rect(), frame, frame.rect(), Qt::NoFormatConversion); //qDebug() << "VIDEO 2" << rect; } frameLock = false; } -#include +/*#include void VideoSurface::resizeEvent(QResizeEvent* event) { // Locks aspect ratio. @@ -196,4 +196,4 @@ void VideoSurface::resizeEvent(QResizeEvent* event) int width = ratio*event->size().width(); setMaximumHeight(width); } -} +}*/ diff --git a/src/video/videosurface.h b/src/video/videosurface.h index 584c39336..a882eb4c6 100644 --- a/src/video/videosurface.h +++ b/src/video/videosurface.h @@ -24,8 +24,9 @@ #include #include #include "src/video/videosource.h" +#include "src/widget/tool/aspectratiowidget.h" -class VideoSurface : public QWidget +class VideoSurface : public AspectRatioWidget { Q_OBJECT @@ -35,10 +36,8 @@ public: ~VideoSurface(); void setSource(VideoSource* src); //NULL is a valid option - QRect getRect() const; - QSize getFrameSize(); - - virtual QSize sizeHint() const override; + //QRect getRect() const; + //QSize getFrameSize() const; signals: void drewNewFrame(); @@ -48,7 +47,7 @@ protected: void unsubscribe(); virtual void paintEvent(QPaintEvent * event) final override; - virtual void resizeEvent(QResizeEvent* event) final override; + //virtual void resizeEvent(QResizeEvent* event) final override; private slots: void onNewFrameAvailable(std::shared_ptr newFrame); diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index d297d769c..be77b6495 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -119,6 +119,8 @@ ChatForm::ChatForm(Friend* chatFriend) retranslateUi(); Translator::registerHandler(std::bind(&ChatForm::retranslateUi, this), this); + + showNetcam(); } ChatForm::~ChatForm() @@ -1098,6 +1100,7 @@ void ChatForm::showNetcam() if (!netcam) netcam = new NetCamView(); + //connect(Widget::getInstance(), &Widget::resized, netcam, &NetCamView::updateSize); netcam->show(Core::getInstance()->getVideoSourceFromCall(callId), f->getDisplayedName()); connect(netcam, &NetCamView::showMessageClicked, this, &ChatForm::onShowMessagesClicked); bodySplitter->insertWidget(0, netcam); diff --git a/src/widget/tool/aspectratiowidget.cpp b/src/widget/tool/aspectratiowidget.cpp new file mode 100644 index 000000000..775d7ddc5 --- /dev/null +++ b/src/widget/tool/aspectratiowidget.cpp @@ -0,0 +1,131 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + qTox is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with qTox. If not, see . +*/ + +#include "aspectratiowidget.h" +#include + +AspectRatioWidget::AspectRatioWidget(QWidget *parent) + : QWidget(parent) + , ratioMode{RespectWidth} + , ratioWidth{QWIDGETSIZE_MAX} +{ + setSizePolicy(QSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum)); + setRatio(1.33f); + //setSizeHint(64, 64 / 1.33f); + size_ = QSize(QWIDGETSIZE_MAX, QWIDGETSIZE_MAX); +} + +float AspectRatioWidget::getRatio() const +{ + return ratio; +} + +bool AspectRatioWidget::aspectRatioRespected() const +{ + int w = static_cast(width() / ratio); + return height() == w - 1 || height() == w || height() == w + 1; +} +#include +QSize AspectRatioWidget::sizeHint() const +{ + //return QSize(QWIDGETSIZE_MAX, QWIDGETSIZE_MAX); + return size_; + switch (ratioMode) + { + case RespectWidth: + return QSize(ratioWidth, width() / ratio); + case RespectHeight: + return QSize(height() * ratio, height()); + case MaximizeSize: + default: + return QSize(ratioWidth, ratioWidth); + } +} +#include +void AspectRatioWidget::setSizeHint(int width, int height) +{ + size_ = QSize(width, height); +} + +void AspectRatioWidget::setMinimum(int minimumHieght) +{ + setMinimumSize(minimumHieght * ratio, minimumHieght); +} + +void AspectRatioWidget::setRatio(float r) +{ + if (ratio == r) + return; + + ratio = r; + setMinimum(minimumHeight()); +} + +void AspectRatioWidget::setRatioMode(RatioMode mode) +{ + ratioMode = mode; + + if (sizeHint().width() > 1) + setMaximumWidth(sizeHint().width()); + + updateGeometry(); +} + +void AspectRatioWidget::setRatioWidth(int w) +{ + ratioWidth = w; +} + +void AspectRatioWidgetList::updateAll() +{ + if (isEmpty()) + return; + + // Maximize the size of all of them, so that they will be equal. + for (AspectRatioWidget* aspectWidget : *this) + { + aspectWidget->setUpdatesEnabled(false); + aspectWidget->setRatioMode(AspectRatioWidget::MaximizeSize); + } + + // Need this to trigger layout recalculations. + qApp->processEvents(QEventLoop::ExcludeUserInputEvents | QEventLoop::ExcludeSocketNotifiers); + + // Now update heights. + for (AspectRatioWidget* aspectWidget : *this) + aspectWidget->setRatioMode(AspectRatioWidget::RespectWidth); + + qApp->processEvents(QEventLoop::ExcludeUserInputEvents); + + // if height does not fit, then decrease width. + // Query only one. Assumes that all of them are next to each other. + if (!at(0)->aspectRatioRespected()) + { + for (AspectRatioWidget* aspectWidget : *this) + aspectWidget->setRatioMode(AspectRatioWidget::RespectHeight); + + qApp->processEvents(QEventLoop::ExcludeUserInputEvents); + } + + for (AspectRatioWidget* aspectWidget : *this) + { + aspectWidget->setUpdatesEnabled(true); + aspectWidget->update(); + } +} diff --git a/src/widget/tool/aspectratiowidget.h b/src/widget/tool/aspectratiowidget.h new file mode 100644 index 000000000..df17595d2 --- /dev/null +++ b/src/widget/tool/aspectratiowidget.h @@ -0,0 +1,66 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + qTox is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with qTox. If not, see . +*/ + +#ifndef ASPECTRATIOWIDGET_H +#define ASPECTRATIOWIDGET_H + +#include + +class AspectRatioWidgetList; + +class AspectRatioWidget : public QWidget +{ + Q_OBJECT +public: + explicit AspectRatioWidget(QWidget *parent = 0); + float getRatio() const; + bool aspectRatioRespected() const; + virtual QSize sizeHint() const final override; + void setSizeHint(int width, int height); + void setMinimum(int minimumHieght); + +protected: + enum RatioMode + { + RespectWidth = 0, + RespectHeight = 1, + MaximizeSize = 2 + }; + + void setRatio(float r); + void setRatioMode(RatioMode mode); + void setRatioWidth(int w = QWIDGETSIZE_MAX); + + friend class AspectRatioWidgetList; + +private: + RatioMode ratioMode; + int ratioWidth; + float ratio; + QSize size_; +}; + +class AspectRatioWidgetList : public QVector +{ +public: + // Must update all simultaneously. + void updateAll(); +}; + +#endif // ASPECTRATIOWIDGET_H diff --git a/src/widget/tool/movablewidget.cpp b/src/widget/tool/movablewidget.cpp index 93d8a74e6..09f46d59b 100644 --- a/src/widget/tool/movablewidget.cpp +++ b/src/widget/tool/movablewidget.cpp @@ -21,33 +21,45 @@ #include #include #include +#include MovableWidget::MovableWidget(QWidget *parent) - : QWidget(parent) + : AspectRatioWidget(parent) { - + setMouseTracking(true); + setRatioWidth(64); + setMinimum(64); + setSizePolicy(QSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored)); + resize(minimumSize()); + actualSize = minimumSize(); } -#include -void MovableWidget::setBoundary(const QRect& boundary, QSize size) + +void MovableWidget::setBoundary(QSize parentSize, QSize oldSize, float xPercent, float yPercent) { - int widthEdge = boundaryRect.x() - width(); - checkBoundaryLeft(widthEdge); - int widthRange = abs(widthEdge) * 2 + boundaryRect.width() - width(); - float xPercent = static_cast(x() - widthEdge) / widthRange; - qDebug() << xPercent << x() - widthEdge << widthRange; - float yPercent = static_cast(y()) / (boundaryRect.height() - height()); + // NOTE: When called, the parentWidget has not resized yet. - if (size.isValid()) - resize(size); + // Prevent division with 0. + if (width() == oldSize.width() || height() == oldSize.height()) + return; - boundaryRect = boundary; - widthEdge = boundaryRect.x() - width(); - checkBoundaryLeft(widthEdge); + float percentageX = x() / static_cast(oldSize.width() - width()); + float percentageY = y() / static_cast(oldSize.height() - height()); - QPoint moveTo = pos(); - moveTo.setX((abs(widthEdge) * 2 + boundaryRect.width() - width()) * xPercent); - moveTo.setY((boundaryRect.height() - height()) * yPercent); - checkBoundary(moveTo); + actualSize.setWidth(actualSize.width() * xPercent); + actualSize.setHeight(actualSize.height() * yPercent); + + if (actualSize.width() == 0) + actualSize.setWidth(1); + + if (actualSize.height() == 0) + actualSize.setHeight(1); + + resize(QSize(round(actualSize.width()), round(actualSize.height()))); + updateGeometry(); + + actualPos = QPointF(percentageX * (parentSize.width() - width()), percentageY * (parentSize.height() - height())); + + QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); move(moveTo); } @@ -55,27 +67,153 @@ void MovableWidget::mousePressEvent(QMouseEvent* event) { if (event->buttons() & Qt::LeftButton) { - moving = true; + if (!(mode & Resize)) + mode |= Moving; + lastPoint = event->globalPos(); } } void MovableWidget::mouseMoveEvent(QMouseEvent* event) { - if (moving) + if (mode & Moving) { QPoint moveTo = pos() - (lastPoint - event->globalPos()); checkBoundary(moveTo); move(moveTo); lastPoint = event->globalPos(); + + actualPos = pos(); + } + else + { + if (!(event->buttons() & Qt::LeftButton)) + { + if (event->x() < 6) + mode |= ResizeLeft; + else + mode &= ~ResizeLeft; + + if (event->y() < 6) + mode |= ResizeUp; + else + mode &= ~ResizeUp; + + if (event->x() > width() - 6) + mode |= ResizeRight; + else + mode &= ~ResizeRight; + + if (event->y() > height() - 6) + mode |= ResizeDown; + else + mode &= ~ResizeDown; + } + + if (mode & Resize) + { + const Modes ResizeUpRight = ResizeUp | ResizeRight; + const Modes ResizeUpLeft = ResizeUp | ResizeLeft; + const Modes ResizeDownRight = ResizeDown | ResizeRight; + const Modes ResizeDownLeft = ResizeDown | ResizeLeft; + + if ((mode & ResizeUpRight) == ResizeUpRight || (mode & ResizeDownLeft) == ResizeDownLeft) + setCursor(Qt::SizeBDiagCursor); + else if ((mode & ResizeUpLeft) == ResizeUpLeft || (mode & ResizeDownRight) == ResizeDownRight) + setCursor(Qt::SizeFDiagCursor); + else if (mode & (ResizeLeft | ResizeRight)) + setCursor(Qt::SizeHorCursor); + else + setCursor(Qt::SizeVerCursor); + + if (event->buttons() & Qt::LeftButton) + { + QPoint lastPosition = pos(); + QPoint displacement = lastPoint - event->globalPos(); + QSize lastSize = size(); + + + if (mode & ResizeUp) + { + lastSize.setHeight(height() + displacement.y()); + + if (lastSize.height() > parentWidget()->height() / 2) + lastPosition.setY(y() - displacement.y() + (lastSize.height() - parentWidget()->height() / 2)); + else + lastPosition.setY(y() - displacement.y()); + } + + if (mode & ResizeLeft) + { + lastSize.setWidth(width() + displacement.x()); + if (lastSize.width() > parentWidget()->width() / 2) + lastPosition.setX(x() - displacement.x() + (lastSize.width() - parentWidget()->width() / 2)); + else + lastPosition.setX(x() - displacement.x()); + } + + if (mode & ResizeRight) + lastSize.setWidth(width() - displacement.x()); + + if (mode & ResizeDown) + lastSize.setHeight(height() - displacement.y()); + + if (lastSize.height() > parentWidget()->height() / 2) + lastSize.setHeight(parentWidget()->height() / 2); + + if (lastSize.width() > parentWidget()->width() / 2) + lastSize.setWidth(parentWidget()->width() / 2); + + if (mode & (ResizeLeft | ResizeRight)) + { + if (mode & (ResizeUp | ResizeDown)) + { + int height = lastSize.width() / getRatio(); + + if (!(mode & ResizeDown)) + lastPosition.setY(lastPosition.y() - (height - lastSize.height())); + + resize(lastSize.width(), height); + + if (lastSize.width() < minimumWidth()) + lastPosition.setX(pos().x()); + + if (height < minimumHeight()) + lastPosition.setY(pos().y()); + } + else + { + resize(lastSize.width(), lastSize.width() / getRatio()); + } + } + else + { + resize(lastSize.height() * getRatio(), lastSize.height()); + } + + updateGeometry(); + + checkBoundary(lastPosition); + + move(lastPosition); + + lastPoint = event->globalPos(); + actualSize = size(); + actualPos = pos(); + } + } + else + { + unsetCursor(); + } } } void MovableWidget::mouseReleaseEvent(QMouseEvent* event) { if (!(event->buttons() & Qt::LeftButton)) - moving = false; + mode = 0; } void MovableWidget::mouseDoubleClickEvent(QMouseEvent* event) @@ -94,12 +232,17 @@ void MovableWidget::mouseDoubleClickEvent(QMouseEvent* event) setGraphicsEffect(nullptr); } } +#include +void MovableWidget::paintEvent(QPaintEvent*) +{ + QPainter painter(this); + painter.setBrush(Qt::black); + painter.drawRect(rect()); + //qDebug() << rect(); +} void MovableWidget::checkBoundary(QPoint& point) const { - if (boundaryRect.isNull()) - return; - int x1, y1, x2, y2; boundaryRect.getCoords(&x1, &y1, &x2, &y2); @@ -109,14 +252,14 @@ void MovableWidget::checkBoundary(QPoint& point) const // Video boundary. - if (point.y() + height() x2) point.setX(x2); if (point.y() > y2) - point.setY(y2); + point.setY(y2);*/ // Parent boundary. if (point.y() < 0) @@ -134,6 +277,6 @@ void MovableWidget::checkBoundaryLeft(int &x) const if (x < 0) x = 0; - if (x + width() < boundaryRect.x()) - x = boundaryRect.x() - width(); + /*if (x + width() < boundaryRect.x()) + x = boundaryRect.x() - width();*/ } diff --git a/src/widget/tool/movablewidget.h b/src/widget/tool/movablewidget.h index 4d6d4920d..1ba7ad904 100644 --- a/src/widget/tool/movablewidget.h +++ b/src/widget/tool/movablewidget.h @@ -21,29 +21,42 @@ #define MOVABLEWIDGET_H #include +#include "aspectratiowidget.h" -class MovableWidget : public QWidget +class MovableWidget : public AspectRatioWidget { public: MovableWidget(QWidget* parent); - void setBoundary(const QRect& boundary, QSize size = QSize()); + void setBoundary(QSize parentSize, QSize oldSize, float xPercent, float yPercent); protected: void mousePressEvent(QMouseEvent* event); void mouseMoveEvent(QMouseEvent* event); void mouseReleaseEvent(QMouseEvent* event); void mouseDoubleClickEvent(QMouseEvent* event); + void paintEvent(QPaintEvent*); private: void checkBoundary(QPoint& point) const; - void checkBoundaryTop(int &x) const; - void checkBoundaryBottom(int &x) const; void checkBoundaryLeft(int &x) const; - void checkBoundaryRight(int &x) const; - bool moving = false; + typedef uint8_t Modes; + + enum Mode : Modes + { + Moving = 0x01, + ResizeLeft = 0x02, + ResizeRight = 0x04, + ResizeUp = 0x08, + ResizeDown = 0x10, + Resize = ResizeLeft | ResizeRight | ResizeUp | ResizeDown + }; + + Modes mode = 0; QPoint lastPoint; QRect boundaryRect; + QSizeF actualSize; + QPointF actualPos; }; #endif // MOVABLEWIDGET_H diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index eb097bb95..f73dc5182 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -210,6 +210,7 @@ void Widget::init() // disable proportional scaling ui->mainSplitter->setStretchFactor(0,0); ui->mainSplitter->setStretchFactor(1,1); + qDebug() << ui->mainSplitter->widget(0)->sizePolicy(); onStatusSet(Status::Offline); From 1522d3848c2484a506231ba8a793481b64efc8c9 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Thu, 13 Aug 2015 09:31:15 -0400 Subject: [PATCH 5/9] Video: Show avatar, clean ups --- qtox.pro | 6 +- src/video/netcamview.cpp | 78 +++++---- src/video/netcamview.h | 4 +- src/video/videosurface.cpp | 218 ++++++++++++++------------ src/video/videosurface.h | 28 ++-- src/widget/form/chatform.cpp | 2 +- src/widget/form/settings/avform.cpp | 2 +- src/widget/tool/aspectratiowidget.cpp | 131 ---------------- src/widget/tool/aspectratiowidget.h | 66 -------- src/widget/tool/movablewidget.cpp | 138 ++++++++++------ src/widget/tool/movablewidget.h | 8 +- src/widget/widget.cpp | 1 - 12 files changed, 284 insertions(+), 398 deletions(-) delete mode 100644 src/widget/tool/aspectratiowidget.cpp delete mode 100644 src/widget/tool/aspectratiowidget.h diff --git a/qtox.pro b/qtox.pro index 28ae6d613..748dfb753 100644 --- a/qtox.pro +++ b/qtox.pro @@ -494,8 +494,7 @@ SOURCES += \ src/widget/friendlistlayout.cpp \ src/widget/genericchatitemlayout.cpp \ src/widget/categorywidget.cpp \ - src/widget/tool/movablewidget.cpp \ - src/widget/tool/aspectratiowidget.cpp + src/widget/tool/movablewidget.cpp HEADERS += \ src/audio/audio.h \ @@ -539,5 +538,4 @@ HEADERS += \ src/widget/friendlistlayout.h \ src/widget/genericchatitemlayout.h \ src/widget/categorywidget.h \ - src/widget/tool/movablewidget.h \ - src/widget/tool/aspectratiowidget.h + src/widget/tool/movablewidget.h diff --git a/src/video/netcamview.cpp b/src/video/netcamview.cpp index 55fb6ec5b..06070449d 100644 --- a/src/video/netcamview.cpp +++ b/src/video/netcamview.cpp @@ -25,8 +25,9 @@ #include #include #include +#include -NetCamView::NetCamView(QWidget* parent) +NetCamView::NetCamView(int friendId, QWidget* parent) : QWidget(parent) , mainLayout(new QHBoxLayout()) , selfFrame{nullptr} @@ -34,38 +35,43 @@ NetCamView::NetCamView(QWidget* parent) QVBoxLayout* layout = new QVBoxLayout(this); setWindowTitle(tr("Tox video")); - videoSurface = new VideoSurface(this); - videoSurface->setSizePolicy(QSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum)); + videoSurface = new VideoSurface(friendId, this); videoSurface->setStyleSheet("background-color: blue;"); - videoSurface->setMinimum(128); + videoSurface->setMinimumHeight(256); + videoSurface->setContentsMargins(6, 6, 6, 6); - mainLayout->addStretch(1); - //QWidget* wid = new QWidget(this); - //wid->setMinimumSize(600, 600); - QVBoxLayout* horLayout = new QVBoxLayout(); - horLayout->addStretch(1); - horLayout->addWidget(videoSurface); - horLayout->addStretch(1); - mainLayout->addLayout(horLayout); + mainLayout->addWidget(videoSurface, 1); selfFrame = new MovableWidget(videoSurface); selfFrame->show(); - selfVideoSurface = new VideoSurface(selfFrame); + selfVideoSurface = new VideoSurface(-1, selfFrame); selfVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); selfVideoSurface->setMouseTracking(true); + selfVideoSurface->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); QHBoxLayout* frameLayout = new QHBoxLayout(selfFrame); frameLayout->addWidget(selfVideoSurface); frameLayout->setMargin(0); - //mainLayout->addWidget(selfVideoSurface); - mainLayout->addStretch(1); - - /*connect(&CameraSource::getInstance(), &CameraSource::deviceOpened, [this]() + connect(&CameraSource::getInstance(), &CameraSource::deviceOpened, [this]() { - connect(selfVideoSurface, &VideoSurface::drewNewFrame, this, &NetCamView::updateSize); - });*/ + //qDebug() << "Device changed"; + //selfFrame->setRatio(selfVideoSurface->getRatio()); + //selfFrame->resetBoundary(); + //connect(selfVideoSurface, &VideoSurface::drewNewFrame, this, &NetCamView::updateSize); + }); + + connect(selfVideoSurface, &VideoSurface::ratioChanged, [this]() + { + qDebug() << "Ratio changed"; + selfFrame->setRatio(selfVideoSurface->getRatio()); + }); + + connect(videoSurface, &VideoSurface::boundaryChanged, [this]() + { + selfFrame->setBoundary(videoSurface->getBoundingRect()); + }); QHBoxLayout* buttonLayout = new QHBoxLayout(); buttonLayout->addStretch(); @@ -76,6 +82,12 @@ NetCamView::NetCamView(QWidget* parent) layout->addLayout(mainLayout, 1); layout->addLayout(buttonLayout); + QFrame* lineFrame = new QFrame(this); + lineFrame->setStyleSheet("border: 1px solid #c1c1c1;"); + lineFrame->setFrameShape(QFrame::HLine); + lineFrame->setMaximumHeight(1); + layout->addWidget(lineFrame); + setShowMessages(false); setStyleSheet("NetCamView { background-color: #c1c1c1; }"); @@ -84,16 +96,14 @@ NetCamView::NetCamView(QWidget* parent) void NetCamView::show(VideoSource *source, const QString &title) { setSource(source); - //selfVideoSurface->setSource(&CameraSource::getInstance()); + selfVideoSurface->setSource(&CameraSource::getInstance()); setTitle(title); QWidget::show(); - //updateSize(); } void NetCamView::hide() { - qDebug() << "jd"; setSource(nullptr); selfVideoSurface->setSource(nullptr); @@ -130,24 +140,22 @@ void NetCamView::setShowMessages(bool show, bool notify) button->setIcon(QIcon()); } } -#include -#include "src/widget/style.h" -void NetCamView::paintEvent(QPaintEvent*) -{ - QPainter painter(this); - painter.setBrush(Style::getColor(Style::ThemeDark)); - painter.drawRect(rect()); -} void NetCamView::resizeEvent(QResizeEvent* event) { QWidget::resizeEvent(event); - updateSize(); + //updateSize(); +} + +void NetCamView::showEvent(QShowEvent *event) +{ + selfFrame->resetBoundary(videoSurface->getBoundingRect()); } void NetCamView::updateSize() { - qDebug() << videoSurface->size(); + qDebug() << selfFrame->geometry(); + /*qDebug() << videoSurface->size(); QSize usableSize = mainLayout->contentsRect().size(); int possibleWidth = usableSize.height() * videoSurface->getRatio(); @@ -165,5 +173,9 @@ void NetCamView::updateSize() QSize newSize = videoSurface->sizeHint(); QSizeF initialSize = initial; - selfFrame->setBoundary(newSize, initial, newSize.width() / initialSize.width(), newSize.height() / initialSize.height()); + selfFrame->setRatio(selfVideoSurface->getRatio());*/ + + //selfFrame->setRatio(selfVideoSurface->getRatio()); + //selfFrame->resetBoundary(videoSurface->getBoundingRect()); + //selfFrame->setBoundary(newSize, initial, newSize.width() / initialSize.width(), newSize.height() / initialSize.height()); } diff --git a/src/video/netcamview.h b/src/video/netcamview.h index 3e610cd1b..c034e81e7 100644 --- a/src/video/netcamview.h +++ b/src/video/netcamview.h @@ -36,7 +36,7 @@ class NetCamView : public QWidget Q_OBJECT public: - NetCamView(QWidget *parent=0); + NetCamView(int friendId, QWidget *parent=0); virtual void show(VideoSource* source, const QString& title); virtual void hide(); @@ -51,8 +51,8 @@ public slots: void setShowMessages(bool show, bool notify = false); protected: - void paintEvent(QPaintEvent* event); void resizeEvent(QResizeEvent* event) final override; + void showEvent(QShowEvent* event) final override; public slots: void updateSize(); diff --git a/src/video/videosurface.cpp b/src/video/videosurface.cpp index f4b308cdb..fd49290a7 100644 --- a/src/video/videosurface.cpp +++ b/src/video/videosurface.cpp @@ -19,22 +19,31 @@ #include "videosurface.h" #include "src/video/videoframe.h" +#include "src/friend.h" +#include "src/friendlist.h" +#include "src/widget/friendwidget.h" +#include "src/persistence/settings.h" #include #include -#include -VideoSurface::VideoSurface(QWidget* parent) - : AspectRatioWidget{parent} + +float getSizeRatio(const QSize size) +{ + return size.width() / static_cast(size.height()); +} + +VideoSurface::VideoSurface(int friendId, QWidget* parent) + : QWidget{parent} , source{nullptr} , frameLock{false} , hasSubscribed{false} + , friendId{friendId} + , ratio{1.0f} { - qDebug() << "ddd" << minimumSize(); - setMinimum(128); - setSizeHint(128, 128); + recalulateBounds(); } -VideoSurface::VideoSurface(VideoSource *source, QWidget* parent) - : VideoSurface(parent) +VideoSurface::VideoSurface(int friendId, VideoSource *source, QWidget* parent) + : VideoSurface(friendId, parent) { setSource(source); } @@ -53,49 +62,19 @@ void VideoSurface::setSource(VideoSource *src) source = src; subscribe(); } -/*#include -QRect VideoSurface::getRect() const +#include +QRect VideoSurface::getBoundingRect() const { - // Fast lock - { - bool expected = false; - while (!frameLock.compare_exchange_weak(expected, true)) - expected = false; - } - - std::shared_ptr last = lastFrame; - frameLock = false; - - if (last) - { - QSize frameSize = lastFrame->getSize(); - QRect rect = this->rect(); - int width = frameSize.width()*rect.height()/frameSize.height(); - rect.setLeft((rect.width()-width)/2); - rect.setWidth(width); - return rect; - } - - return QRect(); + qDebug() << boundingRect.bottomRight() << (QPoint(boundingRect.width(), boundingRect.height()) + boundingRect.topLeft()); + QRect bRect = boundingRect; + bRect.setBottomRight(QPoint(boundingRect.bottom() + 1, boundingRect.right() + 1)); + return boundingRect; } -QSize VideoSurface::getFrameSize() const +float VideoSurface::getRatio() const { - // Fast lock - { - bool expected = false; - while (!frameLock.compare_exchange_weak(expected, true)) - expected = false; - } - - QSize frameSize; - - if (lastFrame) - frameSize = lastFrame->getSize(); - - frameLock = false; - return frameSize; -}*/ + return ratio; +} void VideoSurface::subscribe() { @@ -112,14 +91,14 @@ void VideoSurface::unsubscribe() if (!source || !hasSubscribed) return; - // Fast lock - { - bool expected = false; - while (!frameLock.compare_exchange_weak(expected, true)) - expected = false; - } + lock(); lastFrame.reset(); - frameLock = false; + unlock(); + + ratio = 1.0f; + recalulateBounds(); + emit ratioChanged(); + update(); source->unsubscribe(); hasSubscribed = false; @@ -128,72 +107,105 @@ void VideoSurface::unsubscribe() void VideoSurface::onNewFrameAvailable(std::shared_ptr newFrame) { - // Fast lock + QSize newSize; + + lock(); + lastFrame = newFrame; + newSize = lastFrame->getSize(); + unlock(); + + float newRatio = getSizeRatio(newSize); + + if (newRatio != ratio && isVisible()) { - bool expected = false; - while (!frameLock.compare_exchange_weak(expected, true)) - expected = false; + ratio = newRatio; + recalulateBounds(); + emit ratioChanged(); } - lastFrame = newFrame; - frameLock = false; - setRatio(lastFrame->getSize().width() / static_cast(lastFrame->getSize().height())); - ///setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding); - updateGeometry(); update(); - - emit drewNewFrame(); } - +#include "src/core/core.h" +#include "src/widget/style.h" +#include void VideoSurface::paintEvent(QPaintEvent*) { - // Fast lock - { - bool expected = false; - while (!frameLock.compare_exchange_weak(expected, true)) - expected = false; - } + lock(); QPainter painter(this); - painter.fillRect(painter.viewport(), QColor(193, 193, 193)); + painter.fillRect(painter.viewport(), Qt::black); if (lastFrame) { - /*QSize frameSize = lastFrame->getSize(); - QRect rect = this->rect(); - int width = frameSize.width()*rect.height()/frameSize.height(); - rect.setLeft((rect.width()-width)/2); - rect.setWidth(width);*/ - QImage frame = lastFrame->toQImage(rect().size()); - painter.drawImage(rect(), frame, frame.rect(), Qt::NoFormatConversion); - //qDebug() << "VIDEO 2" << rect; + painter.drawImage(boundingRect, frame, frame.rect(), Qt::NoFormatConversion); } - frameLock = false; + else + { + painter.fillRect(boundingRect, Qt::white); + QPixmap avatar; + + QString userId; + + if (friendId != -1) + userId = FriendList::findFriend(friendId)->getToxId().toString(); + else + userId = Core::getInstance()->getSelfId().toString(); + + avatar = Settings::getInstance().getSavedAvatar(userId); + + if (avatar.isNull()) + avatar = Style::scaleSvgImage(":/img/contact_dark.svg", boundingRect.width(), boundingRect.height()); + + painter.drawPixmap(boundingRect, avatar, avatar.rect()); + } + + unlock(); } -/*#include + void VideoSurface::resizeEvent(QResizeEvent* event) { - // Locks aspect ratio. - QSize frameSize; + QWidget::resizeEvent(event); + recalulateBounds(); +} +void VideoSurface::showEvent(QShowEvent*) +{ + //emit ratioChanged(); +} + +void VideoSurface::recalulateBounds() +{ + if (ratio == 0) + return; + + QPoint pos; + QSize size; + QSize usableSize = contentsRect().size(); + int possibleWidth = usableSize.height() * ratio; + + if (possibleWidth > usableSize.width()) + size = (QSize(usableSize.width(), usableSize.width() / ratio)); + else + size = (QSize(possibleWidth, usableSize.height())); + + pos.setX(width() / 2 - size.width() / 2); + pos.setY(height() / 2 - size.height() / 2); + boundingRect.setRect(pos.x(), pos.y(), size.width(), size.height()); + + emit boundaryChanged(); + + update(); +} + +void VideoSurface::lock() +{ // Fast lock - { - bool expected = false; - while (!frameLock.compare_exchange_weak(expected, true)) - expected = false; - } - - if (lastFrame) - { - frameSize = lastFrame->getSize(); - } + bool expected = false; + while (!frameLock.compare_exchange_weak(expected, true)) + expected = false; +} +void VideoSurface::unlock() +{ frameLock = false; - - if (frameSize.isValid()) - { - float ratio = frameSize.height() / static_cast(frameSize.width()); - int width = ratio*event->size().width(); - setMaximumHeight(width); - } -}*/ +} diff --git a/src/video/videosurface.h b/src/video/videosurface.h index a882eb4c6..f060ce596 100644 --- a/src/video/videosurface.h +++ b/src/video/videosurface.h @@ -24,39 +24,47 @@ #include #include #include "src/video/videosource.h" -#include "src/widget/tool/aspectratiowidget.h" -class VideoSurface : public AspectRatioWidget +class VideoSurface : public QWidget { Q_OBJECT public: - VideoSurface(QWidget* parent=0); - VideoSurface(VideoSource* source, QWidget* parent=0); + VideoSurface(int friendId, QWidget* parent=0); + VideoSurface(int friendId, VideoSource* source, QWidget* parent=0); ~VideoSurface(); void setSource(VideoSource* src); //NULL is a valid option - //QRect getRect() const; - //QSize getFrameSize() const; + QRect getBoundingRect() const; + float getRatio() const; signals: - void drewNewFrame(); + void ratioChanged(); + void boundaryChanged(); protected: void subscribe(); void unsubscribe(); - virtual void paintEvent(QPaintEvent * event) final override; - //virtual void resizeEvent(QResizeEvent* event) final override; + virtual void paintEvent(QPaintEvent* event) final override; + virtual void resizeEvent(QResizeEvent* event) final override; + virtual void showEvent(QShowEvent* event) final override; private slots: void onNewFrameAvailable(std::shared_ptr newFrame); private: + void recalulateBounds(); + void lock(); + void unlock(); + + QRect boundingRect; VideoSource* source; std::shared_ptr lastFrame; - mutable std::atomic_bool frameLock; ///< Fast lock for lastFrame + std::atomic_bool frameLock; ///< Fast lock for lastFrame bool hasSubscribed; + int friendId; + float ratio; }; #endif // SELFCAMVIEW_H diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index be77b6495..adc267ef9 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -1098,7 +1098,7 @@ void ChatForm::SendMessageStr(QString msg) void ChatForm::showNetcam() { if (!netcam) - netcam = new NetCamView(); + netcam = new NetCamView(f->getFriendID()); //connect(Widget::getInstance(), &Widget::resized, netcam, &NetCamView::updateSize); netcam->show(Core::getInstance()->getVideoSourceFromCall(callId), f->getDisplayedName()); diff --git a/src/widget/form/settings/avform.cpp b/src/widget/form/settings/avform.cpp index 475e77251..93761e792 100644 --- a/src/widget/form/settings/avform.cpp +++ b/src/widget/form/settings/avform.cpp @@ -361,7 +361,7 @@ void AVForm::createVideoSurface() { if (camVideoSurface) return; - camVideoSurface = new VideoSurface(bodyUI->CamFrame); + camVideoSurface = new VideoSurface(-1, bodyUI->CamFrame); camVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); camVideoSurface->setMinimumSize(QSize(160, 120)); camVideoSurface->setSource(&camera); diff --git a/src/widget/tool/aspectratiowidget.cpp b/src/widget/tool/aspectratiowidget.cpp deleted file mode 100644 index 775d7ddc5..000000000 --- a/src/widget/tool/aspectratiowidget.cpp +++ /dev/null @@ -1,131 +0,0 @@ -/* - Copyright © 2015 by The qTox Project - - This file is part of qTox, a Qt-based graphical interface for Tox. - - qTox is libre software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - qTox is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with qTox. If not, see . -*/ - -#include "aspectratiowidget.h" -#include - -AspectRatioWidget::AspectRatioWidget(QWidget *parent) - : QWidget(parent) - , ratioMode{RespectWidth} - , ratioWidth{QWIDGETSIZE_MAX} -{ - setSizePolicy(QSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum)); - setRatio(1.33f); - //setSizeHint(64, 64 / 1.33f); - size_ = QSize(QWIDGETSIZE_MAX, QWIDGETSIZE_MAX); -} - -float AspectRatioWidget::getRatio() const -{ - return ratio; -} - -bool AspectRatioWidget::aspectRatioRespected() const -{ - int w = static_cast(width() / ratio); - return height() == w - 1 || height() == w || height() == w + 1; -} -#include -QSize AspectRatioWidget::sizeHint() const -{ - //return QSize(QWIDGETSIZE_MAX, QWIDGETSIZE_MAX); - return size_; - switch (ratioMode) - { - case RespectWidth: - return QSize(ratioWidth, width() / ratio); - case RespectHeight: - return QSize(height() * ratio, height()); - case MaximizeSize: - default: - return QSize(ratioWidth, ratioWidth); - } -} -#include -void AspectRatioWidget::setSizeHint(int width, int height) -{ - size_ = QSize(width, height); -} - -void AspectRatioWidget::setMinimum(int minimumHieght) -{ - setMinimumSize(minimumHieght * ratio, minimumHieght); -} - -void AspectRatioWidget::setRatio(float r) -{ - if (ratio == r) - return; - - ratio = r; - setMinimum(minimumHeight()); -} - -void AspectRatioWidget::setRatioMode(RatioMode mode) -{ - ratioMode = mode; - - if (sizeHint().width() > 1) - setMaximumWidth(sizeHint().width()); - - updateGeometry(); -} - -void AspectRatioWidget::setRatioWidth(int w) -{ - ratioWidth = w; -} - -void AspectRatioWidgetList::updateAll() -{ - if (isEmpty()) - return; - - // Maximize the size of all of them, so that they will be equal. - for (AspectRatioWidget* aspectWidget : *this) - { - aspectWidget->setUpdatesEnabled(false); - aspectWidget->setRatioMode(AspectRatioWidget::MaximizeSize); - } - - // Need this to trigger layout recalculations. - qApp->processEvents(QEventLoop::ExcludeUserInputEvents | QEventLoop::ExcludeSocketNotifiers); - - // Now update heights. - for (AspectRatioWidget* aspectWidget : *this) - aspectWidget->setRatioMode(AspectRatioWidget::RespectWidth); - - qApp->processEvents(QEventLoop::ExcludeUserInputEvents); - - // if height does not fit, then decrease width. - // Query only one. Assumes that all of them are next to each other. - if (!at(0)->aspectRatioRespected()) - { - for (AspectRatioWidget* aspectWidget : *this) - aspectWidget->setRatioMode(AspectRatioWidget::RespectHeight); - - qApp->processEvents(QEventLoop::ExcludeUserInputEvents); - } - - for (AspectRatioWidget* aspectWidget : *this) - { - aspectWidget->setUpdatesEnabled(true); - aspectWidget->update(); - } -} diff --git a/src/widget/tool/aspectratiowidget.h b/src/widget/tool/aspectratiowidget.h deleted file mode 100644 index df17595d2..000000000 --- a/src/widget/tool/aspectratiowidget.h +++ /dev/null @@ -1,66 +0,0 @@ -/* - Copyright © 2015 by The qTox Project - - This file is part of qTox, a Qt-based graphical interface for Tox. - - qTox is libre software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - qTox is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with qTox. If not, see . -*/ - -#ifndef ASPECTRATIOWIDGET_H -#define ASPECTRATIOWIDGET_H - -#include - -class AspectRatioWidgetList; - -class AspectRatioWidget : public QWidget -{ - Q_OBJECT -public: - explicit AspectRatioWidget(QWidget *parent = 0); - float getRatio() const; - bool aspectRatioRespected() const; - virtual QSize sizeHint() const final override; - void setSizeHint(int width, int height); - void setMinimum(int minimumHieght); - -protected: - enum RatioMode - { - RespectWidth = 0, - RespectHeight = 1, - MaximizeSize = 2 - }; - - void setRatio(float r); - void setRatioMode(RatioMode mode); - void setRatioWidth(int w = QWIDGETSIZE_MAX); - - friend class AspectRatioWidgetList; - -private: - RatioMode ratioMode; - int ratioWidth; - float ratio; - QSize size_; -}; - -class AspectRatioWidgetList : public QVector -{ -public: - // Must update all simultaneously. - void updateAll(); -}; - -#endif // ASPECTRATIOWIDGET_H diff --git a/src/widget/tool/movablewidget.cpp b/src/widget/tool/movablewidget.cpp index 09f46d59b..b08f35ac2 100644 --- a/src/widget/tool/movablewidget.cpp +++ b/src/widget/tool/movablewidget.cpp @@ -21,20 +21,26 @@ #include #include #include + #include MovableWidget::MovableWidget(QWidget *parent) - : AspectRatioWidget(parent) + : QWidget(parent) { setMouseTracking(true); - setRatioWidth(64); - setMinimum(64); + setMinimumHeight(64); setSizePolicy(QSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored)); - resize(minimumSize()); actualSize = minimumSize(); + boundaryRect = QRect(0, 0, 0, 0); + setRatio(1.0f); + resize(minimumSize()); + actualPos = QPoint(0, 0); + qDebug() << "SRE" << size(); + + //move() } -void MovableWidget::setBoundary(QSize parentSize, QSize oldSize, float xPercent, float yPercent) +/*void MovableWidget::setBoundary(QSize parentSize, QSize oldSize, float xPercent, float yPercent) { // NOTE: When called, the parentWidget has not resized yet. @@ -61,6 +67,72 @@ void MovableWidget::setBoundary(QSize parentSize, QSize oldSize, float xPercent, QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); move(moveTo); +}*/ + +void MovableWidget::resetBoundary(QRect newBoundary) +{ + boundaryRect = newBoundary; + resize(minimumSize()); + QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); + checkBoundary(moveTo); + move(moveTo); + actualPos = pos(); + actualSize = size(); + qDebug() << "GEO" << geometry(); +} + +void MovableWidget::setBoundary(QRect newBoundary) +{ + if (boundaryRect.isNull()) + { + boundaryRect = newBoundary; + return; + } + + qDebug() << geometry() << actualPos << actualSize; + float changeX = newBoundary.width() / static_cast(boundaryRect.width()); + float changeY = newBoundary.height() / static_cast(boundaryRect.height()); + + float percentageX = (x() - boundaryRect.x()) / static_cast(boundaryRect.width() - width()); + float percentageY = (y() - boundaryRect.y()) / static_cast(boundaryRect.height() - height()); + + actualSize.setWidth(actualSize.width() * changeX); + actualSize.setHeight(actualSize.height() * changeY); + + if (actualSize.width() == 0) + actualSize.setWidth(1); + + if (actualSize.height() == 0) + actualSize.setHeight(1); + + resize(QSize(round(actualSize.width()), round(actualSize.height()))); + + actualPos = QPointF(percentageX * (newBoundary.width() - width()), percentageY * (newBoundary.height() - height())); + actualPos += QPointF(newBoundary.topLeft()); + + QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); + move(moveTo); + + boundaryRect = newBoundary; +} + +float MovableWidget::getRatio() const +{ + return ratio; +} + +void MovableWidget::setRatio(float r) +{ + ratio = r; + setMinimumWidth(minimumHeight() * ratio); + resize(width(), width() / ratio); + QPoint position = QPoint(actualPos.x(), actualPos.y()); + checkBoundary(position); + move(position); + + actualPos = pos(); + actualSize = size(); + qDebug() << "GEO@" << geometry(); } void MovableWidget::mousePressEvent(QMouseEvent* event) @@ -138,8 +210,8 @@ void MovableWidget::mouseMoveEvent(QMouseEvent* event) { lastSize.setHeight(height() + displacement.y()); - if (lastSize.height() > parentWidget()->height() / 2) - lastPosition.setY(y() - displacement.y() + (lastSize.height() - parentWidget()->height() / 2)); + if (lastSize.height() > boundaryRect.height() / 3) + lastPosition.setY(y() - displacement.y() + (lastSize.height() - boundaryRect.height() / 3)); else lastPosition.setY(y() - displacement.y()); } @@ -147,8 +219,8 @@ void MovableWidget::mouseMoveEvent(QMouseEvent* event) if (mode & ResizeLeft) { lastSize.setWidth(width() + displacement.x()); - if (lastSize.width() > parentWidget()->width() / 2) - lastPosition.setX(x() - displacement.x() + (lastSize.width() - parentWidget()->width() / 2)); + if (lastSize.width() > boundaryRect.width() / 3) + lastPosition.setX(x() - displacement.x() + (lastSize.width() - boundaryRect.width() / 3)); else lastPosition.setX(x() - displacement.x()); } @@ -159,11 +231,11 @@ void MovableWidget::mouseMoveEvent(QMouseEvent* event) if (mode & ResizeDown) lastSize.setHeight(height() - displacement.y()); - if (lastSize.height() > parentWidget()->height() / 2) - lastSize.setHeight(parentWidget()->height() / 2); + if (lastSize.height() > boundaryRect.height() / 3) + lastSize.setHeight(boundaryRect.height() / 3); - if (lastSize.width() > parentWidget()->width() / 2) - lastSize.setWidth(parentWidget()->width() / 2); + if (lastSize.width() > boundaryRect.width() / 3) + lastSize.setWidth(boundaryRect.width() / 3); if (mode & (ResizeLeft | ResizeRight)) { @@ -208,6 +280,7 @@ void MovableWidget::mouseMoveEvent(QMouseEvent* event) unsetCursor(); } } + } void MovableWidget::mouseReleaseEvent(QMouseEvent* event) @@ -238,7 +311,6 @@ void MovableWidget::paintEvent(QPaintEvent*) QPainter painter(this); painter.setBrush(Qt::black); painter.drawRect(rect()); - //qDebug() << rect(); } void MovableWidget::checkBoundary(QPoint& point) const @@ -246,37 +318,15 @@ void MovableWidget::checkBoundary(QPoint& point) const int x1, y1, x2, y2; boundaryRect.getCoords(&x1, &y1, &x2, &y2); - x1 = point.x(); - checkBoundaryLeft(x1); - point.setX(x1); + if (point.x() < boundaryRect.left()) + point.setX(boundaryRect.left()); - // Video boundary. + if (point.y() < boundaryRect.top()) + point.setY(boundaryRect.top()); - /*if (point.y() + height() boundaryRect.right() + 1) + point.setX(boundaryRect.right() - width() + 1); - if (point.x() > x2) - point.setX(x2); - - if (point.y() > y2) - point.setY(y2);*/ - - // Parent boundary. - if (point.y() < 0) - point.setY(0); - - if (point.x() + width() > parentWidget()->width()) - point.setX(parentWidget()->width() - width()); - - if (point.y() + height() > parentWidget()->height()) - point.setY(parentWidget()->height() - height()); -} - -void MovableWidget::checkBoundaryLeft(int &x) const -{ - if (x < 0) - x = 0; - - /*if (x + width() < boundaryRect.x()) - x = boundaryRect.x() - width();*/ + if (point.y() + height() > boundaryRect.bottom() + 1) + point.setY(boundaryRect.bottom() - height() + 1); } diff --git a/src/widget/tool/movablewidget.h b/src/widget/tool/movablewidget.h index 1ba7ad904..37c39a349 100644 --- a/src/widget/tool/movablewidget.h +++ b/src/widget/tool/movablewidget.h @@ -23,11 +23,14 @@ #include #include "aspectratiowidget.h" -class MovableWidget : public AspectRatioWidget +class MovableWidget : public QWidget { public: MovableWidget(QWidget* parent); - void setBoundary(QSize parentSize, QSize oldSize, float xPercent, float yPercent); + void resetBoundary(QRect newBoundary); + void setBoundary(QRect newBoundary); + float getRatio() const; + void setRatio(float r); protected: void mousePressEvent(QMouseEvent* event); @@ -57,6 +60,7 @@ private: QRect boundaryRect; QSizeF actualSize; QPointF actualPos; + float ratio; }; #endif // MOVABLEWIDGET_H diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index f73dc5182..eb097bb95 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -210,7 +210,6 @@ void Widget::init() // disable proportional scaling ui->mainSplitter->setStretchFactor(0,0); ui->mainSplitter->setStretchFactor(1,1); - qDebug() << ui->mainSplitter->widget(0)->sizePolicy(); onStatusSet(Status::Offline); From 814bd922ca5bdcdbb164f0d14d886bd46497a75f Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 18 Aug 2015 13:40:11 -0400 Subject: [PATCH 6/9] Video: Group audio view avatars --- qtox.pro | 12 +- src/audio/audio.cpp | 9 +- src/audio/audio.h | 5 +- src/group.cpp | 5 +- src/main.cpp | 2 +- src/video/genericnetcamview.cpp | 69 ++++++++ src/video/genericnetcamview.h | 49 ++++++ src/video/groupnetcamview.cpp | 235 ++++++++++++++++++++++++++++ src/video/groupnetcamview.h | 59 +++++++ src/video/netcamview.cpp | 126 ++++----------- src/video/netcamview.h | 25 +-- src/video/videosurface.cpp | 67 +++++--- src/video/videosurface.h | 8 +- src/widget/form/chatform.cpp | 53 ++----- src/widget/form/chatform.h | 7 +- src/widget/form/genericchatform.cpp | 43 +++++ src/widget/form/genericchatform.h | 7 + src/widget/form/groupchatform.cpp | 45 +++++- src/widget/form/groupchatform.h | 1 + src/widget/tool/flowlayout.cpp | 190 ++++++++++++++++++++++ src/widget/tool/flowlayout.h | 76 +++++++++ src/widget/tool/movablewidget.cpp | 38 +++-- src/widget/tool/movablewidget.h | 1 - src/widget/widget.cpp | 3 +- 24 files changed, 916 insertions(+), 219 deletions(-) create mode 100644 src/video/genericnetcamview.cpp create mode 100644 src/video/genericnetcamview.h create mode 100644 src/video/groupnetcamview.cpp create mode 100644 src/video/groupnetcamview.h create mode 100644 src/widget/tool/flowlayout.cpp create mode 100644 src/widget/tool/flowlayout.h diff --git a/qtox.pro b/qtox.pro index 748dfb753..a8ff3ff8a 100644 --- a/qtox.pro +++ b/qtox.pro @@ -357,7 +357,6 @@ contains(ENABLE_SYSTRAY_GTK_BACKEND, NO) { src/widget/form/loadhistorydialog.h \ src/widget/form/setpassworddialog.h \ src/widget/form/tabcompleter.h \ - src/widget/flowlayout.h \ src/ipc.h \ src/net/autoupdate.h \ src/widget/tool/callconfirmwidget.h \ @@ -383,7 +382,6 @@ contains(ENABLE_SYSTRAY_GTK_BACKEND, NO) { src/widget/form/loadhistorydialog.cpp \ src/widget/form/setpassworddialog.cpp \ src/widget/form/tabcompleter.cpp \ - src/widget/flowlayout.cpp \ src/ipc.cpp \ src/net/autoupdate.cpp \ src/widget/tool/callconfirmwidget.cpp \ @@ -494,7 +492,10 @@ SOURCES += \ src/widget/friendlistlayout.cpp \ src/widget/genericchatitemlayout.cpp \ src/widget/categorywidget.cpp \ - src/widget/tool/movablewidget.cpp + src/widget/tool/movablewidget.cpp \ + src/widget/tool/flowlayout.cpp \ + src/video/genericnetcamview.cpp \ + src/video/groupnetcamview.cpp HEADERS += \ src/audio/audio.h \ @@ -538,4 +539,7 @@ HEADERS += \ src/widget/friendlistlayout.h \ src/widget/genericchatitemlayout.h \ src/widget/categorywidget.h \ - src/widget/tool/movablewidget.h + src/widget/tool/movablewidget.h \ + src/widget/tool/flowlayout.h \ + src/video/genericnetcamview.h \ + src/video/groupnetcamview.h diff --git a/src/audio/audio.cpp b/src/audio/audio.cpp index d206d2c5f..f758636a3 100644 --- a/src/audio/audio.cpp +++ b/src/audio/audio.cpp @@ -270,7 +270,7 @@ void Audio::playMono16Sound(const QByteArray& data) alSourcePlay(alMainSource); alDeleteBuffers(1, &buffer); } - +#include void Audio::playGroupAudioQueued(Tox*,int group, int peer, const int16_t* data, unsigned samples, uint8_t channels, unsigned sample_rate, void* core) { @@ -298,6 +298,13 @@ void Audio::playGroupAudio(int group, int peer, const int16_t* data, alSourcef(call.alSources[peer], AL_GAIN, outputVolume); } + size_t volume = 0;//data[1]; + int bufsize = samples * 2 * channels; + for (int i = 0; i < bufsize; ++i) + volume += abs(data[i]);//std::max(volume, data[i]); + + emit groupAudioPlayed(group, peer, volume / static_cast(bufsize)); + playAudioBuffer(call.alSources[peer], data, samples, channels, sample_rate); } diff --git a/src/audio/audio.h b/src/audio/audio.h index 68f1b7c2e..a81791c73 100644 --- a/src/audio/audio.h +++ b/src/audio/audio.h @@ -41,7 +41,7 @@ class QMutex; struct Tox; class AudioFilterer; -class Audio : QObject +class Audio : public QObject { Q_OBJECT @@ -81,6 +81,9 @@ public slots: void playGroupAudio(int group, int peer, const int16_t* data, unsigned samples, uint8_t channels, unsigned sample_rate); +signals: + void groupAudioPlayed(int group, int peer, unsigned short volume); + private: explicit Audio()=default; ~Audio(); diff --git a/src/group.cpp b/src/group.cpp index a9a9f68a1..028ccf17a 100644 --- a/src/group.cpp +++ b/src/group.cpp @@ -75,12 +75,13 @@ void Group::updatePeer(int peerId, QString name) QString toxid = id.publicKey; peers[peerId] = name; toxids[toxid] = name; - Friend *f = FriendList::findFriend(id); + // I don't know what this is doing here but it changes back to the old name. + /*Friend *f = FriendList::findFriend(id); if (f) { peers[peerId] = f->getDisplayedName(); toxids[toxid] = f->getDisplayedName(); - } + }*/ widget->onUserListChanged(); chatForm->onUserListChanged(); diff --git a/src/main.cpp b/src/main.cpp index 7a493504e..9a982a6f0 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -276,8 +276,8 @@ int main(int argc, char *argv[]) logFile = nullptr; #endif - CameraSource::destroyInstance(); Nexus::destroyInstance(); + CameraSource::destroyInstance(); Settings::destroyInstance(); qDebug() << "Clean exit with status"<. +*/ + +#include "genericnetcamview.h" +#include +#include +#include + +GenericNetCamView::GenericNetCamView(QWidget *parent) + : QWidget(parent) +{ + verLayout = new QVBoxLayout(this); + setWindowTitle(tr("Tox video")); + + int spacing = verLayout->spacing(); + verLayout->setSpacing(0); + + QHBoxLayout* buttonLayout = new QHBoxLayout(); + buttonLayout->addStretch(); + button = new QPushButton(); + buttonLayout->addWidget(button); + connect(button, &QPushButton::clicked, this, &GenericNetCamView::showMessageClicked); + + verLayout->addSpacing(spacing); + verLayout->addLayout(buttonLayout); + verLayout->addSpacing(spacing); + + QFrame* lineFrame = new QFrame(this); + lineFrame->setStyleSheet("border: 1px solid #c1c1c1;"); + lineFrame->setFrameShape(QFrame::HLine); + lineFrame->setMaximumHeight(1); + verLayout->addWidget(lineFrame); + + setShowMessages(false); + + setStyleSheet("NetCamView { background-color: #c1c1c1; }"); +} + +void GenericNetCamView::setShowMessages(bool show, bool notify) +{ + if (show) + { + button->setText(tr("Show Messages")); + + if (notify) + button->setIcon(QIcon("://ui/chatArea/info.svg")); + } + else + { + button->setText(tr("Hide Messages")); + button->setIcon(QIcon()); + } +} diff --git a/src/video/genericnetcamview.h b/src/video/genericnetcamview.h new file mode 100644 index 000000000..05d3bc6bd --- /dev/null +++ b/src/video/genericnetcamview.h @@ -0,0 +1,49 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + qTox is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with qTox. If not, see . +*/ + +#ifndef GENERICNETCAMVIEW_H +#define GENERICNETCAMVIEW_H + +#include + +class VideoSurface; +class QPushButton; +class QVBoxLayout; + +class GenericNetCamView : public QWidget +{ + Q_OBJECT +public: + GenericNetCamView(QWidget* parent); + +signals: + void showMessageClicked(); + +public slots: + void setShowMessages(bool show, bool notify = false); + +protected: + QVBoxLayout* verLayout; + VideoSurface* videoSurface; + +private: + QPushButton* button; +}; + +#endif // GENERICNETCAMVIEW_H diff --git a/src/video/groupnetcamview.cpp b/src/video/groupnetcamview.cpp new file mode 100644 index 000000000..d6d28a0a8 --- /dev/null +++ b/src/video/groupnetcamview.cpp @@ -0,0 +1,235 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + qTox is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with qTox. If not, see . +*/ + +#include "groupnetcamview.h" +#include "src/widget/tool/croppinglabel.h" +#include "src/video/videosurface.h" +#include +#include +#include +#include "src/audio/audio.h" + +#include "src/widget/tool/flowlayout.h" + +class LabeledVideo : public QFrame +{ +public: + LabeledVideo(QWidget* parent = 0, bool expanding = true) + : QFrame(parent) + { + //setFrameStyle(QFrame::Box); + videoSurface = new VideoSurface(-1, 0, expanding); + videoSurface->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); + videoSurface->setMinimumHeight(96); + //videoSurface->setMaximumHeight(96); + connect(videoSurface, &VideoSurface::ratioChanged, this, &LabeledVideo::updateSize); + label = new CroppingLabel(this); + label->setText("Unknown"); + label->setTextFormat(Qt::PlainText); + label->setStyleSheet("color: white"); + //label->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); + //qDebug() << label->sizePolicy(); + label->setAlignment(Qt::AlignCenter); + + QVBoxLayout* layout = new QVBoxLayout(this); + layout->addWidget(videoSurface, 1); + layout->addWidget(label); + + setMouseTracking(true); + } + + ~LabeledVideo() + { + + } + + VideoSurface* getVideoSurface() const + { + return videoSurface; + } + + void setText(const QString& text) + { + label->setText(text); + } + + QString getText() const + { + return label->text(); + } + +protected: + void resizeEvent(QResizeEvent* event) final override + { + QWidget::resizeEvent(event); + updateSize(); + } + + void mousePressEvent(QMouseEvent* event) final override + { + if (videoSurface->isExpanding()) + { + setStyleSheet("QFrame { background-color: #414141; border-radius: 10px; }"); + selected = true; + } + } + +private slots: + void updateSize() + { + if (videoSurface->isExpanding()) + { + int width = videoSurface->height() * videoSurface->getRatio(); + videoSurface->setFixedWidth(width); + setMaximumWidth(width + layout()->margin() * 2); + } + } + +private: + CroppingLabel* label; + VideoSurface* videoSurface; + bool selected = false; +}; + +GroupNetCamView::GroupNetCamView(int group, QWidget *parent) + : GenericNetCamView(parent) + , group(group) +{ + videoLabelSurface = new LabeledVideo(this, false); + videoSurface = videoLabelSurface->getVideoSurface(); + //videoSurface->setExpanding(false); + videoSurface->setMinimumHeight(256); + videoSurface->setContentsMargins(6, 6, 6, 6); + videoLabelSurface->setContentsMargins(0, 0, 0, 0); + videoLabelSurface->layout()->setMargin(0); + videoLabelSurface->setStyleSheet("QFrame { background-color: black; }"); + + verLayout->insertWidget(0, videoLabelSurface, 1); + + QScrollArea* scrollArea = new QScrollArea(); + scrollArea->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + scrollArea->setStyleSheet("QScrollArea { background-color: black; }"); + scrollArea->setFrameStyle(QFrame::NoFrame); + QWidget* widget = new QWidget(nullptr); + scrollArea->setWidget(widget); + scrollArea->setWidgetResizable(true); + horLayout = new QHBoxLayout(widget); + //FlowLayout* horLayout = new FlowLayout(widget); + horLayout->addStretch(); + + LabeledVideo* labeledVideo = new LabeledVideo(this); + horLayout->addWidget(labeledVideo); + horLayout->setAlignment(labeledVideo, Qt::AlignCenter | Qt::AlignHCenter); + + horLayout->addStretch(); + verLayout->insertWidget(1, scrollArea); + scrollArea->setMinimumHeight(labeledVideo->height()); + + connect(&Audio::getInstance(), &Audio::groupAudioPlayed, this, &GroupNetCamView::groupAudioPlayed); + + QTimer* timer = new QTimer(this); + timer->setInterval(1000); + connect(timer, &QTimer::timeout, this, &GroupNetCamView::findActivePeer); + timer->start(); +} + +void GroupNetCamView::clearPeers() +{ + QList keys = videoList.keys(); + + for (int &i : keys) + removePeer(i); +} + +void GroupNetCamView::addPeer(int peer, const QString& name) +{ + LabeledVideo* labeledVideo = new LabeledVideo(this); + labeledVideo->setText(name); + horLayout->insertWidget(horLayout->count() - 1, labeledVideo); + horLayout->setAlignment(labeledVideo, Qt::AlignCenter | Qt::AlignHCenter); + PeerVideo peerVideo; + peerVideo.video = labeledVideo; + videoList.insert(peer, peerVideo); + + findActivePeer(); +} + +void GroupNetCamView::removePeer(int peer) +{ + auto peerVideo = videoList.find(peer); + + if (peerVideo != videoList.end()) + { + LabeledVideo* labeledVideo = peerVideo.value().video; + horLayout->removeWidget(labeledVideo); + labeledVideo->deleteLater(); + videoList.remove(peer); + + findActivePeer(); + } +} +#include +void GroupNetCamView::setActive(int peer) +{ + qDebug() << "HI: " << peer; + if (peer == -1) + { + // Show self. + return; + } + + auto peerVideo = videoList.find(peer); + qDebug() << "BTW" << (peerVideo == videoList.end()); + + if (peerVideo != videoList.end()) + { + // When group video exists: + // videoSurface->setSource(peerVideo.value()->getVideoSurface()->source); + + videoLabelSurface->setText(peerVideo.value().video->getText()); + } +} + +void GroupNetCamView::groupAudioPlayed(int Group, int peer, unsigned short volume) +{ + if (group != Group) + return; + + auto peerVideo = videoList.find(peer); + + if (peerVideo != videoList.end()) + peerVideo.value().volume = volume; +} + +void GroupNetCamView::findActivePeer() +{ + int candidate = -1; + int maximum = 0; + + for (auto peer = videoList.begin(); peer != videoList.end(); ++peer) + { + if (peer.value().volume > maximum) + { + maximum = peer.value().volume; + candidate = peer.key(); + } + } + + setActive(candidate); +} diff --git a/src/video/groupnetcamview.h b/src/video/groupnetcamview.h new file mode 100644 index 000000000..88c3535dd --- /dev/null +++ b/src/video/groupnetcamview.h @@ -0,0 +1,59 @@ +/* + Copyright © 2015 by The qTox Project + + This file is part of qTox, a Qt-based graphical interface for Tox. + + qTox is libre software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + qTox is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with qTox. If not, see . +*/ + +#ifndef GROUPNETCAMVIEW_H +#define GROUPNETCAMVIEW_H + +#include "genericnetcamview.h" +#include + +class LabeledVideo; +class QHBoxLayout; + +class GroupNetCamView : public GenericNetCamView +{ +public: + GroupNetCamView(int group, QWidget* parent = 0); + void clearPeers(); + void addPeer(int peer, const QString &name); + void removePeer(int peer); + +public slots: + void groupAudioPlayed(int group, int peer, unsigned short volume); + +private slots: + void findActivePeer(); + +private: + struct PeerVideo + { + LabeledVideo* video; + unsigned short volume = 0; + }; + + void setActive(int peer); + + QHBoxLayout* horLayout; + QMap videoList; + LabeledVideo* videoLabelSurface; + int activePeer; + int group; +}; + +#endif // GROUPNETCAMVIEW_H diff --git a/src/video/netcamview.cpp b/src/video/netcamview.cpp index 06070449d..36eead001 100644 --- a/src/video/netcamview.cpp +++ b/src/video/netcamview.cpp @@ -1,5 +1,5 @@ /* - Copyright © 2014 by The qTox Project + Copyright © 2014-2015 by The qTox Project This file is part of qTox, a Qt-based graphical interface for Tox. @@ -24,81 +24,55 @@ #include "src/widget/tool/movablewidget.h" #include #include -#include #include NetCamView::NetCamView(int friendId, QWidget* parent) - : QWidget(parent) - , mainLayout(new QHBoxLayout()) + : GenericNetCamView(parent) , selfFrame{nullptr} { - QVBoxLayout* layout = new QVBoxLayout(this); - setWindowTitle(tr("Tox video")); - videoSurface = new VideoSurface(friendId, this); - videoSurface->setStyleSheet("background-color: blue;"); videoSurface->setMinimumHeight(256); videoSurface->setContentsMargins(6, 6, 6, 6); - mainLayout->addWidget(videoSurface, 1); + verLayout->insertWidget(0, videoSurface, 1); - selfFrame = new MovableWidget(videoSurface); - selfFrame->show(); - - selfVideoSurface = new VideoSurface(-1, selfFrame); + selfVideoSurface = new VideoSurface(-1, this, true); selfVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); selfVideoSurface->setMouseTracking(true); selfVideoSurface->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); + selfFrame = new MovableWidget(videoSurface); + selfFrame->show(); + QHBoxLayout* frameLayout = new QHBoxLayout(selfFrame); frameLayout->addWidget(selfVideoSurface); frameLayout->setMargin(0); - connect(&CameraSource::getInstance(), &CameraSource::deviceOpened, [this]() - { - //qDebug() << "Device changed"; - //selfFrame->setRatio(selfVideoSurface->getRatio()); - //selfFrame->resetBoundary(); - //connect(selfVideoSurface, &VideoSurface::drewNewFrame, this, &NetCamView::updateSize); - }); - - connect(selfVideoSurface, &VideoSurface::ratioChanged, [this]() - { - qDebug() << "Ratio changed"; - selfFrame->setRatio(selfVideoSurface->getRatio()); - }); + updateRatio(); + connect(selfVideoSurface, &VideoSurface::ratioChanged, this, &NetCamView::updateRatio); connect(videoSurface, &VideoSurface::boundaryChanged, [this]() { - selfFrame->setBoundary(videoSurface->getBoundingRect()); + QRect boundingRect = videoSurface->getBoundingRect(); + updateFrameSize(boundingRect.size()); + selfFrame->setBoundary(boundingRect); }); - QHBoxLayout* buttonLayout = new QHBoxLayout(); - buttonLayout->addStretch(); - button = new QPushButton(); - buttonLayout->addWidget(button); - connect(button, &QPushButton::clicked, this, &NetCamView::showMessageClicked); - - layout->addLayout(mainLayout, 1); - layout->addLayout(buttonLayout); - - QFrame* lineFrame = new QFrame(this); - lineFrame->setStyleSheet("border: 1px solid #c1c1c1;"); - lineFrame->setFrameShape(QFrame::HLine); - lineFrame->setMaximumHeight(1); - layout->addWidget(lineFrame); - - setShowMessages(false); - - setStyleSheet("NetCamView { background-color: #c1c1c1; }"); + connect(videoSurface, &VideoSurface::ratioChanged, [this]() + { + selfFrame->setMinimumWidth(selfFrame->minimumHeight() * selfVideoSurface->getRatio()); + QRect boundingRect = videoSurface->getBoundingRect(); + updateFrameSize(boundingRect.size()); + selfFrame->resetBoundary(boundingRect); + }); } void NetCamView::show(VideoSource *source, const QString &title) { setSource(source); selfVideoSurface->setSource(&CameraSource::getInstance()); - setTitle(title); + setTitle(title); QWidget::show(); } @@ -125,57 +99,23 @@ void NetCamView::setTitle(const QString &title) setWindowTitle(title); } -void NetCamView::setShowMessages(bool show, bool notify) -{ - if (show) - { - button->setText(tr("Show Messages")); - - if (notify) - button->setIcon(QIcon("://ui/chatArea/info.svg")); - } - else - { - button->setText(tr("Hide Messages")); - button->setIcon(QIcon()); - } -} - -void NetCamView::resizeEvent(QResizeEvent* event) -{ - QWidget::resizeEvent(event); - //updateSize(); -} - void NetCamView::showEvent(QShowEvent *event) { selfFrame->resetBoundary(videoSurface->getBoundingRect()); } -void NetCamView::updateSize() +void NetCamView::updateRatio() { - qDebug() << selfFrame->geometry(); - /*qDebug() << videoSurface->size(); - QSize usableSize = mainLayout->contentsRect().size(); - int possibleWidth = usableSize.height() * videoSurface->getRatio(); - - QSize initial = videoSurface->sizeHint(); - - if (!initial.isValid()) - return; - - if (possibleWidth > usableSize.width()) - videoSurface->setSizeHint(usableSize.width(), usableSize.width() / videoSurface->getRatio()); - else - videoSurface->setSizeHint(usableSize.height() * videoSurface->getRatio(), usableSize.height()); - - videoSurface->updateGeometry(); - - QSize newSize = videoSurface->sizeHint(); - QSizeF initialSize = initial; - selfFrame->setRatio(selfVideoSurface->getRatio());*/ - - //selfFrame->setRatio(selfVideoSurface->getRatio()); - //selfFrame->resetBoundary(videoSurface->getBoundingRect()); - //selfFrame->setBoundary(newSize, initial, newSize.width() / initialSize.width(), newSize.height() / initialSize.height()); + selfFrame->setMinimumWidth(selfFrame->minimumHeight() * selfVideoSurface->getRatio()); + selfFrame->setRatio(selfVideoSurface->getRatio()); +} + +void NetCamView::updateFrameSize(QSize size) +{ + selfFrame->setMaximumSize(size.height() / 3, size.width() / 3); + + if (selfFrame->maximumWidth() > selfFrame->maximumHeight()) + selfFrame->setMaximumWidth(selfFrame->maximumHeight() * selfVideoSurface->getRatio()); + else + selfFrame->setMaximumHeight(selfFrame->maximumWidth() / selfVideoSurface->getRatio()); } diff --git a/src/video/netcamview.h b/src/video/netcamview.h index c034e81e7..0015b1755 100644 --- a/src/video/netcamview.h +++ b/src/video/netcamview.h @@ -1,5 +1,5 @@ /* - Copyright © 2014 by The qTox Project + Copyright © 2014-2015 by The qTox Project This file is part of qTox, a Qt-based graphical interface for Tox. @@ -20,18 +20,15 @@ #ifndef NETCAMVIEW_H #define NETCAMVIEW_H -#include -#include "src/widget/tool/aspectratiowidget.h" +#include "genericnetcamview.h" class QHBoxLayout; struct vpx_image; -class VideoSurface; class VideoSource; class QFrame; class MovableWidget; -class QPushButton; -class NetCamView : public QWidget +class NetCamView : public GenericNetCamView { Q_OBJECT @@ -44,27 +41,17 @@ public: void setSource(VideoSource* s); void setTitle(const QString& title); -signals: - void showMessageClicked(); - -public slots: - void setShowMessages(bool show, bool notify = false); - protected: - void resizeEvent(QResizeEvent* event) final override; void showEvent(QShowEvent* event) final override; -public slots: - void updateSize(); +private slots: + void updateRatio(); private: - void updateFrameSize(); + void updateFrameSize(QSize size); - QHBoxLayout* mainLayout; - VideoSurface* videoSurface; VideoSurface* selfVideoSurface; MovableWidget* selfFrame; - QPushButton* button; bool e = false; }; diff --git a/src/video/videosurface.cpp b/src/video/videosurface.cpp index fd49290a7..29dc4f3e4 100644 --- a/src/video/videosurface.cpp +++ b/src/video/videosurface.cpp @@ -25,19 +25,21 @@ #include "src/persistence/settings.h" #include #include +#include float getSizeRatio(const QSize size) { return size.width() / static_cast(size.height()); } -VideoSurface::VideoSurface(int friendId, QWidget* parent) +VideoSurface::VideoSurface(int friendId, QWidget* parent, bool expanding) : QWidget{parent} , source{nullptr} , frameLock{false} - , hasSubscribed{false} + , hasSubscribed{0} , friendId{friendId} , ratio{1.0f} + , expanding{expanding} { recalulateBounds(); } @@ -53,6 +55,11 @@ VideoSurface::~VideoSurface() unsubscribe(); } +bool VideoSurface::isExpanding() const +{ + return expanding; +} + void VideoSurface::setSource(VideoSource *src) { if (source == src) @@ -62,10 +69,9 @@ void VideoSurface::setSource(VideoSource *src) source = src; subscribe(); } -#include + QRect VideoSurface::getBoundingRect() const { - qDebug() << boundingRect.bottomRight() << (QPoint(boundingRect.width(), boundingRect.height()) + boundingRect.topLeft()); QRect bRect = boundingRect; bRect.setBottomRight(QPoint(boundingRect.bottom() + 1, boundingRect.right() + 1)); return boundingRect; @@ -78,17 +84,23 @@ float VideoSurface::getRatio() const void VideoSurface::subscribe() { - if (source && !hasSubscribed) + assert(hasSubscribed >= 0); + + if (source && hasSubscribed++ == 0) { source->subscribe(); - hasSubscribed = true; connect(source, &VideoSource::frameAvailable, this, &VideoSurface::onNewFrameAvailable); } } void VideoSurface::unsubscribe() { - if (!source || !hasSubscribed) + assert(hasSubscribed >= 0); + + if (!source || hasSubscribed == 0) + return; + + if (--hasSubscribed != 0) return; lock(); @@ -98,10 +110,9 @@ void VideoSurface::unsubscribe() ratio = 1.0f; recalulateBounds(); emit ratioChanged(); - update(); + emit boundaryChanged(); source->unsubscribe(); - hasSubscribed = false; disconnect(source, &VideoSource::frameAvailable, this, &VideoSurface::onNewFrameAvailable); } @@ -121,6 +132,7 @@ void VideoSurface::onNewFrameAvailable(std::shared_ptr newFrame) ratio = newRatio; recalulateBounds(); emit ratioChanged(); + emit boundaryChanged(); } update(); @@ -166,33 +178,38 @@ void VideoSurface::resizeEvent(QResizeEvent* event) { QWidget::resizeEvent(event); recalulateBounds(); + emit boundaryChanged(); } void VideoSurface::showEvent(QShowEvent*) { //emit ratioChanged(); } - +#include void VideoSurface::recalulateBounds() { - if (ratio == 0) - return; - - QPoint pos; - QSize size; - QSize usableSize = contentsRect().size(); - int possibleWidth = usableSize.height() * ratio; - - if (possibleWidth > usableSize.width()) - size = (QSize(usableSize.width(), usableSize.width() / ratio)); + if (expanding) + { + boundingRect = contentsRect(); + } else - size = (QSize(possibleWidth, usableSize.height())); + { + QPoint pos; + QSize size; + QSize usableSize = contentsRect().size(); + int possibleWidth = usableSize.height() * ratio; - pos.setX(width() / 2 - size.width() / 2); - pos.setY(height() / 2 - size.height() / 2); - boundingRect.setRect(pos.x(), pos.y(), size.width(), size.height()); + if (possibleWidth > usableSize.width()) + size = (QSize(usableSize.width(), usableSize.width() / ratio)); + else + size = (QSize(possibleWidth, usableSize.height())); - emit boundaryChanged(); + pos.setX(width() / 2 - size.width() / 2); + pos.setY(height() / 2 - size.height() / 2); + boundingRect.setRect(pos.x(), pos.y(), size.width(), size.height()); + } + + qDebug() << contentsRect(); update(); } diff --git a/src/video/videosurface.h b/src/video/videosurface.h index f060ce596..029287359 100644 --- a/src/video/videosurface.h +++ b/src/video/videosurface.h @@ -30,10 +30,11 @@ class VideoSurface : public QWidget Q_OBJECT public: - VideoSurface(int friendId, QWidget* parent=0); - VideoSurface(int friendId, VideoSource* source, QWidget* parent=0); + VideoSurface(int friendId, QWidget* parent = 0, bool expanding = false); + VideoSurface(int friendId, VideoSource* source, QWidget* parent = 0); ~VideoSurface(); + bool isExpanding() const; void setSource(VideoSource* src); //NULL is a valid option QRect getBoundingRect() const; float getRatio() const; @@ -62,9 +63,10 @@ private: VideoSource* source; std::shared_ptr lastFrame; std::atomic_bool frameLock; ///< Fast lock for lastFrame - bool hasSubscribed; + uint8_t hasSubscribed; int friendId; float ratio; + bool expanding; }; #endif // SELFCAMVIEW_H diff --git a/src/widget/form/chatform.cpp b/src/widget/form/chatform.cpp index adc267ef9..56941c7ca 100644 --- a/src/widget/form/chatform.cpp +++ b/src/widget/form/chatform.cpp @@ -42,7 +42,6 @@ #include "src/core/cstring.h" #include "src/widget/tool/callconfirmwidget.h" #include "src/widget/friendwidget.h" -#include "src/video/netcamview.h" #include "src/widget/form/loadhistorydialog.h" #include "src/widget/tool/chattextedit.h" #include "src/widget/widget.h" @@ -53,6 +52,7 @@ #include "src/chatlog/chatlinecontentproxy.h" #include "src/chatlog/content/text.h" #include "src/chatlog/chatlog.h" +#include "src/video/netcamview.h" #include "src/persistence/offlinemsgengine.h" #include "src/widget/tool/screenshotgrabber.h" #include "src/widget/tool/flyoutoverlaywidget.h" @@ -77,7 +77,6 @@ ChatForm::ChatForm(Friend* chatFriend) typingTimer.setSingleShot(true); - netcam = nullptr; callDurationTimer = nullptr; disableCallButtonsTimer = nullptr; @@ -90,7 +89,6 @@ ChatForm::ChatForm(Friend* chatFriend) callDuration->hide(); chatWidget->setMinimumHeight(160); - connect(bodySplitter, &QSplitter::splitterMoved, this, &ChatForm::onSplitterMoved); connect(this, &GenericChatForm::messageInserted, this, &ChatForm::onMessageInserted); loadHistoryAction = menu.addAction(QString(), this, SLOT(onLoadHistory())); @@ -120,7 +118,7 @@ ChatForm::ChatForm(Friend* chatFriend) retranslateUi(); Translator::registerHandler(std::bind(&ChatForm::retranslateUi, this), this); - showNetcam(); + //showNetcam(); } ChatForm::~ChatForm() @@ -751,6 +749,13 @@ void ChatForm::onAvatarChange(uint32_t FriendId, const QPixmap &pic) avatar->setPixmap(pic); } +GenericNetCamView *ChatForm::createNetcam() +{ + NetCamView* view = new NetCamView(f->getFriendID(), this); + view->show(Core::getInstance()->getVideoSourceFromCall(callId), f->getDisplayedName()); + return view; +} + void ChatForm::dragEnterEvent(QDragEnterEvent *ev) { if (ev->mimeData()->hasUrls()) @@ -938,31 +943,12 @@ void ChatForm::onLoadHistory() } } -void ChatForm::onSplitterMoved(int, int) -{ - if (netcam) - netcam->setShowMessages(bodySplitter->sizes()[1] == 0); -} - void ChatForm::onMessageInserted() { if (netcam && bodySplitter->sizes()[1] == 0) netcam->setShowMessages(true, true); } -void ChatForm::onShowMessagesClicked() -{ - if (netcam) - { - if (bodySplitter->sizes()[1] == 0) - bodySplitter->setSizes({1, 1}); - else - bodySplitter->setSizes({1, 0}); - - onSplitterMoved(0, 0); - } -} - void ChatForm::startCounter() { if (!callDurationTimer) @@ -1095,27 +1081,6 @@ void ChatForm::SendMessageStr(QString msg) } } -void ChatForm::showNetcam() -{ - if (!netcam) - netcam = new NetCamView(f->getFriendID()); - - //connect(Widget::getInstance(), &Widget::resized, netcam, &NetCamView::updateSize); - netcam->show(Core::getInstance()->getVideoSourceFromCall(callId), f->getDisplayedName()); - connect(netcam, &NetCamView::showMessageClicked, this, &ChatForm::onShowMessagesClicked); - bodySplitter->insertWidget(0, netcam); - bodySplitter->setCollapsible(0, false); -} - -void ChatForm::hideNetcam() -{ - if (!netcam) - return; - netcam->hide(); - delete netcam; - netcam = nullptr; -} - void ChatForm::retranslateUi() { loadHistoryAction->setText(tr("Load chat history...")); diff --git a/src/widget/form/chatform.h b/src/widget/form/chatform.h index 63e03702e..b3ecd24ef 100644 --- a/src/widget/form/chatform.h +++ b/src/widget/form/chatform.h @@ -30,7 +30,6 @@ class Friend; class FileTransferInstance; -class NetCamView; class QPixmap; class CallConfirmWidget; class QHideEvent; @@ -100,16 +99,13 @@ private slots: void onScreenshotClicked(); void onScreenshotTaken(const QPixmap &pixmap); void doScreenshot(); - void onSplitterMoved(int pos, int index); void onMessageInserted(); - void onShowMessagesClicked(); private: void retranslateUi(); protected: - void showNetcam(); - void hideNetcam(); + virtual GenericNetCamView* createNetcam() final override; // drag & drop virtual void dragEnterEvent(QDragEnterEvent* ev) final override; virtual void dropEvent(QDropEvent* ev) final override; @@ -119,7 +115,6 @@ protected: private: Friend* f; CroppingLabel *statusMessageLabel; - NetCamView* netcam; int callId; QLabel *callDuration; QTimer *callDurationTimer; diff --git a/src/widget/form/genericchatform.cpp b/src/widget/form/genericchatform.cpp index 8db137390..20c4df831 100644 --- a/src/widget/form/genericchatform.cpp +++ b/src/widget/form/genericchatform.cpp @@ -43,6 +43,7 @@ #include "src/chatlog/content/timestamp.h" #include "src/widget/tool/flyoutoverlaywidget.h" #include "src/widget/translator.h" +#include "src/video/genericnetcamview.h" GenericChatForm::GenericChatForm(QWidget *parent) : QWidget(parent) @@ -128,6 +129,7 @@ GenericChatForm::GenericChatForm(QWidget *parent) setLayout(mainLayout); bodySplitter = new QSplitter(Qt::Vertical, this); + connect(bodySplitter, &QSplitter::splitterMoved, this, &GenericChatForm::onSplitterMoved); QWidget* contentWidget = new QWidget(this); QVBoxLayout* contentLayout = new QVBoxLayout(contentWidget); @@ -201,6 +203,8 @@ GenericChatForm::GenericChatForm(QWidget *parent) retranslateUi(); Translator::registerHandler(std::bind(&GenericChatForm::retranslateUi, this), this); + + netcam = nullptr; } GenericChatForm::~GenericChatForm() @@ -518,6 +522,25 @@ bool GenericChatForm::eventFilter(QObject* object, QEvent* event) return false; } +void GenericChatForm::onSplitterMoved(int, int) +{ + if (netcam) + netcam->setShowMessages(bodySplitter->sizes()[1] == 0); +} + +void GenericChatForm::onShowMessagesClicked() +{ + if (netcam) + { + if (bodySplitter->sizes()[1] == 0) + bodySplitter->setSizes({1, 1}); + else + bodySplitter->setSizes({1, 0}); + + onSplitterMoved(0, 0); + } +} + void GenericChatForm::retranslateUi() { sendButton->setToolTip(tr("Send message")); @@ -529,3 +552,23 @@ void GenericChatForm::retranslateUi() saveChatAction->setText(tr("Save chat log")); clearAction->setText(tr("Clear displayed messages")); } + +void GenericChatForm::showNetcam() +{ + if (!netcam) + netcam = createNetcam(); + + connect(netcam, &GenericNetCamView::showMessageClicked, this, &GenericChatForm::onShowMessagesClicked); + + bodySplitter->insertWidget(0, netcam); + bodySplitter->setCollapsible(0, false); +} + +void GenericChatForm::hideNetcam() +{ + if (!netcam) + return; + netcam->hide(); + delete netcam; + netcam = nullptr; +} diff --git a/src/widget/form/genericchatform.h b/src/widget/form/genericchatform.h index 7b835e687..3df92c464 100644 --- a/src/widget/form/genericchatform.h +++ b/src/widget/form/genericchatform.h @@ -41,6 +41,7 @@ class MaskablePixmapWidget; class Widget; class FlyoutOverlayWidget; class QSplitter; +class GenericNetCamView; namespace Ui { class MainWindow; @@ -87,11 +88,16 @@ protected slots: void onSelectAllClicked(); void showFileMenu(); void hideFileMenu(); + void onShowMessagesClicked(); + void onSplitterMoved(int pos, int index); private: void retranslateUi(); protected: + void showNetcam(); + void hideNetcam(); + virtual GenericNetCamView* createNetcam() = 0; QString resolveToxId(const ToxId &id); void insertChatMessage(ChatMessage::Ptr msg); void adjustFileMenuPosition(); @@ -122,6 +128,7 @@ protected: bool audioInputFlag; bool audioOutputFlag; QSplitter* bodySplitter; + GenericNetCamView* netcam; }; #endif // GENERICCHATFORM_H diff --git a/src/widget/form/groupchatform.cpp b/src/widget/form/groupchatform.cpp index 4638cd114..88a2d40d9 100644 --- a/src/widget/form/groupchatform.cpp +++ b/src/widget/form/groupchatform.cpp @@ -29,6 +29,7 @@ #include "src/persistence/historykeeper.h" #include "src/widget/flowlayout.h" #include "src/widget/translator.h" +#include "src/video/groupnetcamview.h" #include #include #include @@ -191,7 +192,14 @@ void GroupChatForm::onUserListChanged() orderizer[names[i]] = peerLabels[i]; if (group->isSelfPeerNumber(i)) peerLabels[i]->setStyleSheet("QLabel {color : green;}"); + + if (netcam && !group->isSelfPeerNumber(i)) + static_cast(netcam)->addPeer(i, names[i]); } + + if (netcam) + static_cast(netcam)->clearPeers(); + // now alphabetize and add to layout names.sort(Qt::CaseInsensitive); for (unsigned i=0; isetSingleShot(true); - connect(peerAudioTimers[peer], &QTimer::timeout, [=]{this->peerLabels[peer]->setStyleSheet(""); - delete this->peerAudioTimers[peer]; - this->peerAudioTimers[peer] = nullptr;}); + connect(peerAudioTimers[peer], &QTimer::timeout, [this, peer] + { + if (netcam) + static_cast(netcam)->removePeer(peer); + + if (peer >= peerLabels.size()) + return; + + peerLabels[peer]->setStyleSheet(""); + delete peerAudioTimers[peer]; + peerAudioTimers[peer] = nullptr; + }); + + if (netcam) + { + static_cast(netcam)->removePeer(peer); + static_cast(netcam)->addPeer(peer, group->getPeerList()[peer]); + } } peerAudioTimers[peer]->start(500); } @@ -308,6 +331,7 @@ void GroupChatForm::onCallClicked() volButton->style()->polish(volButton); volButton->setToolTip(tr("Mute call")); inCall = true; + showNetcam(); } else { @@ -324,9 +348,24 @@ void GroupChatForm::onCallClicked() volButton->style()->polish(volButton); volButton->setToolTip(""); inCall = false; + hideNetcam(); } } +GenericNetCamView *GroupChatForm::createNetcam() +{ + GroupNetCamView* view = new GroupNetCamView(group->getGroupId(), this); + + QStringList names = group->getPeerList(); + for (unsigned i=0; iisSelfPeerNumber(i)) + static_cast(view)->addPeer(i, names[i]); + } + + return view; +} + void GroupChatForm::keyPressEvent(QKeyEvent* ev) { // Push to talk (CTRL+P) diff --git a/src/widget/form/groupchatform.h b/src/widget/form/groupchatform.h index c3b13e3e9..73eaeed5c 100644 --- a/src/widget/form/groupchatform.h +++ b/src/widget/form/groupchatform.h @@ -49,6 +49,7 @@ private slots: void onCallClicked(); protected: + virtual GenericNetCamView* createNetcam() final override; virtual void keyPressEvent(QKeyEvent* ev) final override; virtual void keyReleaseEvent(QKeyEvent* ev) final override; // drag & drop diff --git a/src/widget/tool/flowlayout.cpp b/src/widget/tool/flowlayout.cpp new file mode 100644 index 000000000..727410432 --- /dev/null +++ b/src/widget/tool/flowlayout.cpp @@ -0,0 +1,190 @@ +/**************************************************************************** +** +** Copyright (C) 2015 The Qt Company Ltd. +** Contact: http://www.qt.io/licensing/ +** +** This file is part of the examples of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of The Qt Company Ltd nor the names of its +** contributors may be used to endorse or promote products derived +** from this software without specific prior written permission. +** +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#include + +#include "flowlayout.h" +FlowLayout::FlowLayout(QWidget *parent, int margin, int hSpacing, int vSpacing) + : QLayout(parent), m_hSpace(hSpacing), m_vSpace(vSpacing) +{ + setContentsMargins(margin, margin, margin, margin); +} + +FlowLayout::FlowLayout(int margin, int hSpacing, int vSpacing) + : m_hSpace(hSpacing), m_vSpace(vSpacing) +{ + setContentsMargins(margin, margin, margin, margin); +} + +FlowLayout::~FlowLayout() +{ + QLayoutItem *item; + while ((item = takeAt(0))) + delete item; +} + +void FlowLayout::addItem(QLayoutItem *item) +{ + itemList.append(item); +} + +int FlowLayout::horizontalSpacing() const +{ + if (m_hSpace >= 0) { + return m_hSpace; + } else { + return smartSpacing(QStyle::PM_LayoutHorizontalSpacing); + } +} + +int FlowLayout::verticalSpacing() const +{ + if (m_vSpace >= 0) { + return m_vSpace; + } else { + return smartSpacing(QStyle::PM_LayoutVerticalSpacing); + } +} + +int FlowLayout::count() const +{ + return itemList.size(); +} + +QLayoutItem *FlowLayout::itemAt(int index) const +{ + return itemList.value(index); +} + +QLayoutItem *FlowLayout::takeAt(int index) +{ + if (index >= 0 && index < itemList.size()) + return itemList.takeAt(index); + else + return 0; +} + +Qt::Orientations FlowLayout::expandingDirections() const +{ + return 0; +} + +bool FlowLayout::hasHeightForWidth() const +{ + return true; +} + +int FlowLayout::heightForWidth(int width) const +{ + int height = doLayout(QRect(0, 0, width, 0), true); + return height; +} + +void FlowLayout::setGeometry(const QRect &rect) +{ + QLayout::setGeometry(rect); + doLayout(rect, false); +} + +QSize FlowLayout::sizeHint() const +{ + return minimumSize(); +} + +QSize FlowLayout::minimumSize() const +{ + QSize size; + QLayoutItem *item; + foreach (item, itemList) + size = size.expandedTo(item->minimumSize()); + + size += QSize(2*margin(), 2*margin()); + return size; +} + +int FlowLayout::doLayout(const QRect &rect, bool testOnly) const +{ + int left, top, right, bottom; + getContentsMargins(&left, &top, &right, &bottom); + QRect effectiveRect = rect.adjusted(+left, +top, -right, -bottom); + int x = effectiveRect.x(); + int y = effectiveRect.y(); + int lineHeight = 0; + + QLayoutItem *item; + foreach (item, itemList) { + QWidget *wid = item->widget(); + int spaceX = horizontalSpacing(); + if (spaceX == -1) + spaceX = wid->style()->layoutSpacing( + QSizePolicy::PushButton, QSizePolicy::PushButton, Qt::Horizontal); + int spaceY = verticalSpacing(); + if (spaceY == -1) + spaceY = wid->style()->layoutSpacing( + QSizePolicy::PushButton, QSizePolicy::PushButton, Qt::Vertical); + int nextX = x + item->sizeHint().width() + spaceX; + if (nextX - spaceX > effectiveRect.right() && lineHeight > 0) { + x = effectiveRect.x(); + y = y + lineHeight + spaceY; + nextX = x + item->sizeHint().width() + spaceX; + lineHeight = 0; + } + + if (!testOnly) + item->setGeometry(QRect(QPoint(x, y), item->sizeHint())); + + x = nextX; + lineHeight = qMax(lineHeight, item->sizeHint().height()); + } + return y + lineHeight - rect.y() + bottom; +} + +int FlowLayout::smartSpacing(QStyle::PixelMetric pm) const +{ + QObject *parent = this->parent(); + if (!parent) { + return -1; + } else if (parent->isWidgetType()) { + QWidget *pw = static_cast(parent); + return pw->style()->pixelMetric(pm, 0, pw); + } else { + return static_cast(parent)->spacing(); + } +} diff --git a/src/widget/tool/flowlayout.h b/src/widget/tool/flowlayout.h new file mode 100644 index 000000000..6e61ef9cf --- /dev/null +++ b/src/widget/tool/flowlayout.h @@ -0,0 +1,76 @@ +/**************************************************************************** +** +** Copyright (C) 2015 The Qt Company Ltd. +** Contact: http://www.qt.io/licensing/ +** +** This file is part of the examples of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of The Qt Company Ltd nor the names of its +** contributors may be used to endorse or promote products derived +** from this software without specific prior written permission. +** +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#ifndef FLOWLAYOUT_H +#define FLOWLAYOUT_H + +#include +#include +#include +class FlowLayout : public QLayout +{ +public: + explicit FlowLayout(QWidget *parent, int margin = -1, int hSpacing = -1, int vSpacing = -1); + explicit FlowLayout(int margin = -1, int hSpacing = -1, int vSpacing = -1); + ~FlowLayout(); + + void addItem(QLayoutItem *item) Q_DECL_OVERRIDE; + int horizontalSpacing() const; + int verticalSpacing() const; + Qt::Orientations expandingDirections() const Q_DECL_OVERRIDE; + bool hasHeightForWidth() const Q_DECL_OVERRIDE; + int heightForWidth(int) const Q_DECL_OVERRIDE; + int count() const Q_DECL_OVERRIDE; + QLayoutItem *itemAt(int index) const Q_DECL_OVERRIDE; + QSize minimumSize() const Q_DECL_OVERRIDE; + void setGeometry(const QRect &rect) Q_DECL_OVERRIDE; + QSize sizeHint() const Q_DECL_OVERRIDE; + QLayoutItem *takeAt(int index) Q_DECL_OVERRIDE; + +private: + int doLayout(const QRect &rect, bool testOnly) const; + int smartSpacing(QStyle::PixelMetric pm) const; + + QList itemList; + int m_hSpace; + int m_vSpace; +}; + +#endif // FLOWLAYOUT_H diff --git a/src/widget/tool/movablewidget.cpp b/src/widget/tool/movablewidget.cpp index b08f35ac2..4fd770181 100644 --- a/src/widget/tool/movablewidget.cpp +++ b/src/widget/tool/movablewidget.cpp @@ -24,6 +24,11 @@ #include +static float getNetcamX = 0.0f; +static float getNetcamY = 0.0f; +static float getNetcamW = 0.0f; +static float getNetcamH = 0.0f; + MovableWidget::MovableWidget(QWidget *parent) : QWidget(parent) { @@ -35,7 +40,6 @@ MovableWidget::MovableWidget(QWidget *parent) setRatio(1.0f); resize(minimumSize()); actualPos = QPoint(0, 0); - qDebug() << "SRE" << size(); //move() } @@ -72,13 +76,17 @@ MovableWidget::MovableWidget(QWidget *parent) void MovableWidget::resetBoundary(QRect newBoundary) { boundaryRect = newBoundary; - resize(minimumSize()); + + //actualSize = QSize(getNetcamW * newBoundary.width(), getNetcamH * newBoundary.height()); + resize(QSize(round(actualSize.width()), round(actualSize.height()))); + + //actualPos = QPointF(getNetcamX * (newBoundary.width() - width()), getNetcamY * (newBoundary.height() - height())); QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); checkBoundary(moveTo); move(moveTo); - actualPos = pos(); - actualSize = size(); - qDebug() << "GEO" << geometry(); + actualPos = moveTo; + + qDebug() << "GGGG" << geometry(); } void MovableWidget::setBoundary(QRect newBoundary) @@ -89,7 +97,6 @@ void MovableWidget::setBoundary(QRect newBoundary) return; } - qDebug() << geometry() << actualPos << actualSize; float changeX = newBoundary.width() / static_cast(boundaryRect.width()); float changeY = newBoundary.height() / static_cast(boundaryRect.height()); @@ -113,7 +120,9 @@ void MovableWidget::setBoundary(QRect newBoundary) QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); move(moveTo); + qDebug() << "hhhhh" << geometry() << boundaryRect << newBoundary; boundaryRect = newBoundary; + } float MovableWidget::getRatio() const @@ -124,7 +133,6 @@ float MovableWidget::getRatio() const void MovableWidget::setRatio(float r) { ratio = r; - setMinimumWidth(minimumHeight() * ratio); resize(width(), width() / ratio); QPoint position = QPoint(actualPos.x(), actualPos.y()); checkBoundary(position); @@ -210,8 +218,8 @@ void MovableWidget::mouseMoveEvent(QMouseEvent* event) { lastSize.setHeight(height() + displacement.y()); - if (lastSize.height() > boundaryRect.height() / 3) - lastPosition.setY(y() - displacement.y() + (lastSize.height() - boundaryRect.height() / 3)); + if (lastSize.height() > maximumHeight()) + lastPosition.setY(y() - displacement.y() + (lastSize.height() - maximumHeight())); else lastPosition.setY(y() - displacement.y()); } @@ -219,8 +227,8 @@ void MovableWidget::mouseMoveEvent(QMouseEvent* event) if (mode & ResizeLeft) { lastSize.setWidth(width() + displacement.x()); - if (lastSize.width() > boundaryRect.width() / 3) - lastPosition.setX(x() - displacement.x() + (lastSize.width() - boundaryRect.width() / 3)); + if (lastSize.width() > maximumWidth()) + lastPosition.setX(x() - displacement.x() + (lastSize.width() - maximumWidth())); else lastPosition.setX(x() - displacement.x()); } @@ -231,11 +239,11 @@ void MovableWidget::mouseMoveEvent(QMouseEvent* event) if (mode & ResizeDown) lastSize.setHeight(height() - displacement.y()); - if (lastSize.height() > boundaryRect.height() / 3) - lastSize.setHeight(boundaryRect.height() / 3); + if (lastSize.height() > maximumHeight()) + lastSize.setHeight(maximumHeight()); - if (lastSize.width() > boundaryRect.width() / 3) - lastSize.setWidth(boundaryRect.width() / 3); + if (lastSize.width() > maximumWidth()) + lastSize.setWidth(maximumWidth()); if (mode & (ResizeLeft | ResizeRight)) { diff --git a/src/widget/tool/movablewidget.h b/src/widget/tool/movablewidget.h index 37c39a349..86f1ef072 100644 --- a/src/widget/tool/movablewidget.h +++ b/src/widget/tool/movablewidget.h @@ -21,7 +21,6 @@ #define MOVABLEWIDGET_H #include -#include "aspectratiowidget.h" class MovableWidget : public QWidget { diff --git a/src/widget/widget.cpp b/src/widget/widget.cpp index eb097bb95..a902d83f8 100644 --- a/src/widget/widget.cpp +++ b/src/widget/widget.cpp @@ -1059,7 +1059,8 @@ void Widget::onGroupNamelistChanged(int groupnumber, int peernumber, uint8_t Cha } else if (change == TOX_CHAT_CHANGE_PEER_NAME) // core overwrites old name before telling us it changed... { - g->updatePeer(peernumber,Nexus::getCore()->getGroupPeerName(groupnumber, peernumber)); + qDebug() << "UPDATING PEER"; + g->updatePeer(peernumber, name); } } From 50041a31406643dca03afce9dbb8ccba07ed271d Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Tue, 18 Aug 2015 17:44:34 -0400 Subject: [PATCH 7/9] Video: Initial video when call starts --- qtox.pro | 3 +- src/persistence/settings.cpp | 15 +++ src/persistence/settings.h | 4 + src/video/groupnetcamview.cpp | 32 +++-- src/video/groupnetcamview.h | 1 + src/video/netcamview.cpp | 15 ++- src/video/videosurface.cpp | 39 +++--- src/video/videosurface.h | 8 +- src/widget/form/settings/avform.cpp | 14 +- src/widget/form/settings/avform.h | 2 +- src/widget/tool/flowlayout.cpp | 190 ---------------------------- src/widget/tool/flowlayout.h | 76 ----------- 12 files changed, 89 insertions(+), 310 deletions(-) delete mode 100644 src/widget/tool/flowlayout.cpp delete mode 100644 src/widget/tool/flowlayout.h diff --git a/qtox.pro b/qtox.pro index a8ff3ff8a..a5833ee86 100644 --- a/qtox.pro +++ b/qtox.pro @@ -383,6 +383,7 @@ contains(ENABLE_SYSTRAY_GTK_BACKEND, NO) { src/widget/form/setpassworddialog.cpp \ src/widget/form/tabcompleter.cpp \ src/ipc.cpp \ + src/widget/flowlayout.cpp \ src/net/autoupdate.cpp \ src/widget/tool/callconfirmwidget.cpp \ src/widget/systemtrayicon.cpp \ @@ -493,7 +494,6 @@ SOURCES += \ src/widget/genericchatitemlayout.cpp \ src/widget/categorywidget.cpp \ src/widget/tool/movablewidget.cpp \ - src/widget/tool/flowlayout.cpp \ src/video/genericnetcamview.cpp \ src/video/groupnetcamview.cpp @@ -540,6 +540,5 @@ HEADERS += \ src/widget/genericchatitemlayout.h \ src/widget/categorywidget.h \ src/widget/tool/movablewidget.h \ - src/widget/tool/flowlayout.h \ src/video/genericnetcamview.h \ src/video/groupnetcamview.h diff --git a/src/persistence/settings.cpp b/src/persistence/settings.cpp index 9a4935c1e..96296ad6d 100644 --- a/src/persistence/settings.cpp +++ b/src/persistence/settings.cpp @@ -227,6 +227,7 @@ void Settings::loadGlobal() s.beginGroup("Video"); videoDev = s.value("videoDev", "").toString(); camVideoRes = s.value("camVideoRes",QSize()).toSize(); + camVideoFPS = s.value("camVideoFPS", 0).toUInt(); s.endGroup(); // Read the embedded DHT bootsrap nodes list if needed @@ -422,6 +423,7 @@ void Settings::saveGlobal() s.beginGroup("Video"); s.setValue("videoDev", videoDev); s.setValue("camVideoRes",camVideoRes); + s.setValue("camVideoFPS",camVideoFPS); s.endGroup(); } @@ -1175,6 +1177,19 @@ void Settings::setCamVideoRes(QSize newValue) camVideoRes = newValue; } +unsigned short Settings::getCamVideoFPS() const +{ + QMutexLocker locker{&bigLock}; + return camVideoFPS; +} + +void Settings::setCamVideoFPS(unsigned short newValue) +{ + QMutexLocker locker{&bigLock}; + camVideoFPS = newValue; +} + + QString Settings::getFriendAdress(const QString &publicKey) const { QMutexLocker locker{&bigLock}; diff --git a/src/persistence/settings.h b/src/persistence/settings.h index cd9ca5715..41b23338f 100644 --- a/src/persistence/settings.h +++ b/src/persistence/settings.h @@ -163,6 +163,9 @@ public: QSize getCamVideoRes() const; void setCamVideoRes(QSize newValue); + unsigned short getCamVideoFPS() const; + void setCamVideoFPS(unsigned short newValue); + bool isAnimationEnabled() const; void setAnimationEnabled(bool newValue); @@ -356,6 +359,7 @@ private: // Video QString videoDev; QSize camVideoRes; + unsigned short camVideoFPS; struct friendProp { diff --git a/src/video/groupnetcamview.cpp b/src/video/groupnetcamview.cpp index d6d28a0a8..06e2dbcce 100644 --- a/src/video/groupnetcamview.cpp +++ b/src/video/groupnetcamview.cpp @@ -24,8 +24,8 @@ #include #include #include "src/audio/audio.h" - -#include "src/widget/tool/flowlayout.h" +#include "src/core/core.h" +#include class LabeledVideo : public QFrame { @@ -34,7 +34,7 @@ public: : QFrame(parent) { //setFrameStyle(QFrame::Box); - videoSurface = new VideoSurface(-1, 0, expanding); + videoSurface = new VideoSurface(QPixmap(), 0, expanding); videoSurface->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); videoSurface->setMinimumHeight(96); //videoSurface->setMaximumHeight(96); @@ -133,13 +133,13 @@ GroupNetCamView::GroupNetCamView(int group, QWidget *parent) //FlowLayout* horLayout = new FlowLayout(widget); horLayout->addStretch(); - LabeledVideo* labeledVideo = new LabeledVideo(this); - horLayout->addWidget(labeledVideo); - horLayout->setAlignment(labeledVideo, Qt::AlignCenter | Qt::AlignHCenter); + selfVideoSurface = new LabeledVideo(this); + horLayout->addWidget(selfVideoSurface); + horLayout->setAlignment(selfVideoSurface, Qt::AlignCenter | Qt::AlignHCenter); horLayout->addStretch(); verLayout->insertWidget(1, scrollArea); - scrollArea->setMinimumHeight(labeledVideo->height()); + scrollArea->setMinimumHeight(selfVideoSurface->height()); connect(&Audio::getInstance(), &Audio::groupAudioPlayed, this, &GroupNetCamView::groupAudioPlayed); @@ -147,6 +147,18 @@ GroupNetCamView::GroupNetCamView(int group, QWidget *parent) timer->setInterval(1000); connect(timer, &QTimer::timeout, this, &GroupNetCamView::findActivePeer); timer->start(); + + connect(Core::getInstance(), &Core::selfAvatarChanged, [this]() + { + selfVideoSurface->update(); + findActivePeer(); + }); + connect(Core::getInstance(), &Core::usernameSet, [this](const QString& username) + { + selfVideoSurface->setText(username); + findActivePeer(); + }); + selfVideoSurface->setText(Core::getInstance()->getUsername()); } void GroupNetCamView::clearPeers() @@ -184,18 +196,16 @@ void GroupNetCamView::removePeer(int peer) findActivePeer(); } } -#include + void GroupNetCamView::setActive(int peer) { - qDebug() << "HI: " << peer; if (peer == -1) { - // Show self. + videoLabelSurface->setText(selfVideoSurface->getText()); return; } auto peerVideo = videoList.find(peer); - qDebug() << "BTW" << (peerVideo == videoList.end()); if (peerVideo != videoList.end()) { diff --git a/src/video/groupnetcamview.h b/src/video/groupnetcamview.h index 88c3535dd..0994f8c2c 100644 --- a/src/video/groupnetcamview.h +++ b/src/video/groupnetcamview.h @@ -52,6 +52,7 @@ private: QHBoxLayout* horLayout; QMap videoList; LabeledVideo* videoLabelSurface; + LabeledVideo* selfVideoSurface; int activePeer; int group; }; diff --git a/src/video/netcamview.cpp b/src/video/netcamview.cpp index 36eead001..5dbf83115 100644 --- a/src/video/netcamview.cpp +++ b/src/video/netcamview.cpp @@ -19,9 +19,12 @@ #include "netcamview.h" #include "camerasource.h" +#include "src/friend.h" +#include "src/friendlist.h" #include "src/core/core.h" #include "src/video/videosurface.h" #include "src/widget/tool/movablewidget.h" +#include "src/persistence/settings.h" #include #include #include @@ -30,13 +33,14 @@ NetCamView::NetCamView(int friendId, QWidget* parent) : GenericNetCamView(parent) , selfFrame{nullptr} { - videoSurface = new VideoSurface(friendId, this); + QString id = FriendList::findFriend(friendId)->getToxId().toString(); + videoSurface = new VideoSurface(Settings::getInstance().getSavedAvatar(id), this); videoSurface->setMinimumHeight(256); videoSurface->setContentsMargins(6, 6, 6, 6); verLayout->insertWidget(0, videoSurface, 1); - selfVideoSurface = new VideoSurface(-1, this, true); + selfVideoSurface = new VideoSurface(Settings::getInstance().getSavedAvatar(Core::getInstance()->getSelfId().toString()), this, true); selfVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); selfVideoSurface->setMouseTracking(true); selfVideoSurface->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); @@ -65,6 +69,13 @@ NetCamView::NetCamView(int friendId, QWidget* parent) updateFrameSize(boundingRect.size()); selfFrame->resetBoundary(boundingRect); }); + + VideoMode videoMode; + QSize videoSize = Settings::getInstance().getCamVideoRes(); + videoMode.width = videoSize.width(); + videoMode.height = videoSize.height(); + videoMode.FPS = Settings::getInstance().getCamVideoFPS(); + CameraSource::getInstance().open(Settings::getInstance().getVideoDev(), videoMode); } void NetCamView::show(VideoSource *source, const QString &title) diff --git a/src/video/videosurface.cpp b/src/video/videosurface.cpp index 29dc4f3e4..1402d3366 100644 --- a/src/video/videosurface.cpp +++ b/src/video/videosurface.cpp @@ -32,20 +32,20 @@ float getSizeRatio(const QSize size) return size.width() / static_cast(size.height()); } -VideoSurface::VideoSurface(int friendId, QWidget* parent, bool expanding) +VideoSurface::VideoSurface(const QPixmap& avatar, QWidget* parent, bool expanding) : QWidget{parent} , source{nullptr} , frameLock{false} , hasSubscribed{0} - , friendId{friendId} + , avatar{avatar} , ratio{1.0f} , expanding{expanding} { recalulateBounds(); } -VideoSurface::VideoSurface(int friendId, VideoSource *source, QWidget* parent) - : VideoSurface(friendId, parent) +VideoSurface::VideoSurface(const QPixmap& avatar, VideoSource *source, QWidget* parent) + : VideoSurface(avatar, parent) { setSource(source); } @@ -82,6 +82,16 @@ float VideoSurface::getRatio() const return ratio; } +void VideoSurface::setAvatar(const QPixmap &pixmap) +{ + avatar = pixmap; +} + +QPixmap VideoSurface::getAvatar() const +{ + return avatar; +} + void VideoSurface::subscribe() { assert(hasSubscribed >= 0); @@ -154,21 +164,12 @@ void VideoSurface::paintEvent(QPaintEvent*) else { painter.fillRect(boundingRect, Qt::white); - QPixmap avatar; + QPixmap drawnAvatar = avatar; - QString userId; + if (drawnAvatar.isNull()) + drawnAvatar = Style::scaleSvgImage(":/img/contact_dark.svg", boundingRect.width(), boundingRect.height()); - if (friendId != -1) - userId = FriendList::findFriend(friendId)->getToxId().toString(); - else - userId = Core::getInstance()->getSelfId().toString(); - - avatar = Settings::getInstance().getSavedAvatar(userId); - - if (avatar.isNull()) - avatar = Style::scaleSvgImage(":/img/contact_dark.svg", boundingRect.width(), boundingRect.height()); - - painter.drawPixmap(boundingRect, avatar, avatar.rect()); + painter.drawPixmap(boundingRect, drawnAvatar, drawnAvatar.rect()); } unlock(); @@ -185,7 +186,7 @@ void VideoSurface::showEvent(QShowEvent*) { //emit ratioChanged(); } -#include + void VideoSurface::recalulateBounds() { if (expanding) @@ -209,8 +210,6 @@ void VideoSurface::recalulateBounds() boundingRect.setRect(pos.x(), pos.y(), size.width(), size.height()); } - qDebug() << contentsRect(); - update(); } diff --git a/src/video/videosurface.h b/src/video/videosurface.h index 029287359..533776b14 100644 --- a/src/video/videosurface.h +++ b/src/video/videosurface.h @@ -30,14 +30,16 @@ class VideoSurface : public QWidget Q_OBJECT public: - VideoSurface(int friendId, QWidget* parent = 0, bool expanding = false); - VideoSurface(int friendId, VideoSource* source, QWidget* parent = 0); + VideoSurface(const QPixmap& avatar, QWidget* parent = 0, bool expanding = false); + VideoSurface(const QPixmap& avatar, VideoSource* source, QWidget* parent = 0); ~VideoSurface(); bool isExpanding() const; void setSource(VideoSource* src); //NULL is a valid option QRect getBoundingRect() const; float getRatio() const; + void setAvatar(const QPixmap& pixmap); + QPixmap getAvatar() const; signals: void ratioChanged(); @@ -64,7 +66,7 @@ private: std::shared_ptr lastFrame; std::atomic_bool frameLock; ///< Fast lock for lastFrame uint8_t hasSubscribed; - int friendId; + QPixmap avatar; float ratio; bool expanding; }; diff --git a/src/widget/form/settings/avform.cpp b/src/widget/form/settings/avform.cpp index 93761e792..45e07374c 100644 --- a/src/widget/form/settings/avform.cpp +++ b/src/widget/form/settings/avform.cpp @@ -102,6 +102,7 @@ void AVForm::on_videoModescomboBox_currentIndexChanged(int index) QString devName = videoDeviceList[devIndex].first; VideoMode mode = videoModes[index]; Settings::getInstance().setCamVideoRes(QSize(mode.width, mode.height)); + Settings::getInstance().setCamVideoFPS(mode.FPS); camera.open(devName, mode); } @@ -123,10 +124,11 @@ void AVForm::updateVideoModes(int curIndex) bodyUI->videoModescomboBox->clear(); int prefResIndex = -1; QSize prefRes = Settings::getInstance().getCamVideoRes(); + unsigned short prefFPS = Settings::getInstance().getCamVideoFPS(); for (int i=0; ivideoModescomboBox->blockSignals(previouslyBlocked); camera.open(dev); + killVideoSurface(); + createVideoSurface(); } -void AVForm::onResProbingFinished(QList res) +/*void AVForm::onResProbingFinished(QList res) { QSize savedRes = Settings::getInstance().getCamVideoRes(); int savedResIndex = -1; @@ -215,7 +219,7 @@ void AVForm::onResProbingFinished(QList res) bodyUI->videoModescomboBox->setCurrentIndex(savedResIndex); else bodyUI->videoModescomboBox->setCurrentIndex(bodyUI->videoModescomboBox->count()-1); -} +}*/ void AVForm::hideEvent(QHideEvent *) { @@ -242,7 +246,7 @@ void AVForm::getVideoDevices() videoDevIndex = bodyUI->videoDevCombobox->count()-1; } //addItem changes currentIndex -> reset - bodyUI->videoDevCombobox->setCurrentIndex(-1); + //bodyUI->videoDevCombobox->setCurrentIndex(-1); bodyUI->videoDevCombobox->setCurrentIndex(videoDevIndex); bodyUI->videoDevCombobox->blockSignals(false); updateVideoModes(videoDevIndex); @@ -361,7 +365,7 @@ void AVForm::createVideoSurface() { if (camVideoSurface) return; - camVideoSurface = new VideoSurface(-1, bodyUI->CamFrame); + camVideoSurface = new VideoSurface(QPixmap(), bodyUI->CamFrame); camVideoSurface->setObjectName(QStringLiteral("CamVideoSurface")); camVideoSurface->setMinimumSize(QSize(160, 120)); camVideoSurface->setSource(&camera); diff --git a/src/widget/form/settings/avform.h b/src/widget/form/settings/avform.h index 63fcd5580..d4c1a748f 100644 --- a/src/widget/form/settings/avform.h +++ b/src/widget/form/settings/avform.h @@ -63,7 +63,7 @@ private slots: // camera void onVideoDevChanged(int index); - void onResProbingFinished(QList res); + //void onResProbingFinished(QList res); virtual void hideEvent(QHideEvent*) final override; virtual void showEvent(QShowEvent*) final override; diff --git a/src/widget/tool/flowlayout.cpp b/src/widget/tool/flowlayout.cpp deleted file mode 100644 index 727410432..000000000 --- a/src/widget/tool/flowlayout.cpp +++ /dev/null @@ -1,190 +0,0 @@ -/**************************************************************************** -** -** Copyright (C) 2015 The Qt Company Ltd. -** Contact: http://www.qt.io/licensing/ -** -** This file is part of the examples of the Qt Toolkit. -** -** $QT_BEGIN_LICENSE:BSD$ -** You may use this file under the terms of the BSD license as follows: -** -** "Redistribution and use in source and binary forms, with or without -** modification, are permitted provided that the following conditions are -** met: -** * Redistributions of source code must retain the above copyright -** notice, this list of conditions and the following disclaimer. -** * Redistributions in binary form must reproduce the above copyright -** notice, this list of conditions and the following disclaimer in -** the documentation and/or other materials provided with the -** distribution. -** * Neither the name of The Qt Company Ltd nor the names of its -** contributors may be used to endorse or promote products derived -** from this software without specific prior written permission. -** -** -** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS -** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT -** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT -** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, -** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT -** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, -** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY -** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT -** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE -** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." -** -** $QT_END_LICENSE$ -** -****************************************************************************/ - -#include - -#include "flowlayout.h" -FlowLayout::FlowLayout(QWidget *parent, int margin, int hSpacing, int vSpacing) - : QLayout(parent), m_hSpace(hSpacing), m_vSpace(vSpacing) -{ - setContentsMargins(margin, margin, margin, margin); -} - -FlowLayout::FlowLayout(int margin, int hSpacing, int vSpacing) - : m_hSpace(hSpacing), m_vSpace(vSpacing) -{ - setContentsMargins(margin, margin, margin, margin); -} - -FlowLayout::~FlowLayout() -{ - QLayoutItem *item; - while ((item = takeAt(0))) - delete item; -} - -void FlowLayout::addItem(QLayoutItem *item) -{ - itemList.append(item); -} - -int FlowLayout::horizontalSpacing() const -{ - if (m_hSpace >= 0) { - return m_hSpace; - } else { - return smartSpacing(QStyle::PM_LayoutHorizontalSpacing); - } -} - -int FlowLayout::verticalSpacing() const -{ - if (m_vSpace >= 0) { - return m_vSpace; - } else { - return smartSpacing(QStyle::PM_LayoutVerticalSpacing); - } -} - -int FlowLayout::count() const -{ - return itemList.size(); -} - -QLayoutItem *FlowLayout::itemAt(int index) const -{ - return itemList.value(index); -} - -QLayoutItem *FlowLayout::takeAt(int index) -{ - if (index >= 0 && index < itemList.size()) - return itemList.takeAt(index); - else - return 0; -} - -Qt::Orientations FlowLayout::expandingDirections() const -{ - return 0; -} - -bool FlowLayout::hasHeightForWidth() const -{ - return true; -} - -int FlowLayout::heightForWidth(int width) const -{ - int height = doLayout(QRect(0, 0, width, 0), true); - return height; -} - -void FlowLayout::setGeometry(const QRect &rect) -{ - QLayout::setGeometry(rect); - doLayout(rect, false); -} - -QSize FlowLayout::sizeHint() const -{ - return minimumSize(); -} - -QSize FlowLayout::minimumSize() const -{ - QSize size; - QLayoutItem *item; - foreach (item, itemList) - size = size.expandedTo(item->minimumSize()); - - size += QSize(2*margin(), 2*margin()); - return size; -} - -int FlowLayout::doLayout(const QRect &rect, bool testOnly) const -{ - int left, top, right, bottom; - getContentsMargins(&left, &top, &right, &bottom); - QRect effectiveRect = rect.adjusted(+left, +top, -right, -bottom); - int x = effectiveRect.x(); - int y = effectiveRect.y(); - int lineHeight = 0; - - QLayoutItem *item; - foreach (item, itemList) { - QWidget *wid = item->widget(); - int spaceX = horizontalSpacing(); - if (spaceX == -1) - spaceX = wid->style()->layoutSpacing( - QSizePolicy::PushButton, QSizePolicy::PushButton, Qt::Horizontal); - int spaceY = verticalSpacing(); - if (spaceY == -1) - spaceY = wid->style()->layoutSpacing( - QSizePolicy::PushButton, QSizePolicy::PushButton, Qt::Vertical); - int nextX = x + item->sizeHint().width() + spaceX; - if (nextX - spaceX > effectiveRect.right() && lineHeight > 0) { - x = effectiveRect.x(); - y = y + lineHeight + spaceY; - nextX = x + item->sizeHint().width() + spaceX; - lineHeight = 0; - } - - if (!testOnly) - item->setGeometry(QRect(QPoint(x, y), item->sizeHint())); - - x = nextX; - lineHeight = qMax(lineHeight, item->sizeHint().height()); - } - return y + lineHeight - rect.y() + bottom; -} - -int FlowLayout::smartSpacing(QStyle::PixelMetric pm) const -{ - QObject *parent = this->parent(); - if (!parent) { - return -1; - } else if (parent->isWidgetType()) { - QWidget *pw = static_cast(parent); - return pw->style()->pixelMetric(pm, 0, pw); - } else { - return static_cast(parent)->spacing(); - } -} diff --git a/src/widget/tool/flowlayout.h b/src/widget/tool/flowlayout.h deleted file mode 100644 index 6e61ef9cf..000000000 --- a/src/widget/tool/flowlayout.h +++ /dev/null @@ -1,76 +0,0 @@ -/**************************************************************************** -** -** Copyright (C) 2015 The Qt Company Ltd. -** Contact: http://www.qt.io/licensing/ -** -** This file is part of the examples of the Qt Toolkit. -** -** $QT_BEGIN_LICENSE:BSD$ -** You may use this file under the terms of the BSD license as follows: -** -** "Redistribution and use in source and binary forms, with or without -** modification, are permitted provided that the following conditions are -** met: -** * Redistributions of source code must retain the above copyright -** notice, this list of conditions and the following disclaimer. -** * Redistributions in binary form must reproduce the above copyright -** notice, this list of conditions and the following disclaimer in -** the documentation and/or other materials provided with the -** distribution. -** * Neither the name of The Qt Company Ltd nor the names of its -** contributors may be used to endorse or promote products derived -** from this software without specific prior written permission. -** -** -** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS -** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT -** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR -** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT -** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, -** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT -** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, -** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY -** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT -** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE -** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." -** -** $QT_END_LICENSE$ -** -****************************************************************************/ - -#ifndef FLOWLAYOUT_H -#define FLOWLAYOUT_H - -#include -#include -#include -class FlowLayout : public QLayout -{ -public: - explicit FlowLayout(QWidget *parent, int margin = -1, int hSpacing = -1, int vSpacing = -1); - explicit FlowLayout(int margin = -1, int hSpacing = -1, int vSpacing = -1); - ~FlowLayout(); - - void addItem(QLayoutItem *item) Q_DECL_OVERRIDE; - int horizontalSpacing() const; - int verticalSpacing() const; - Qt::Orientations expandingDirections() const Q_DECL_OVERRIDE; - bool hasHeightForWidth() const Q_DECL_OVERRIDE; - int heightForWidth(int) const Q_DECL_OVERRIDE; - int count() const Q_DECL_OVERRIDE; - QLayoutItem *itemAt(int index) const Q_DECL_OVERRIDE; - QSize minimumSize() const Q_DECL_OVERRIDE; - void setGeometry(const QRect &rect) Q_DECL_OVERRIDE; - QSize sizeHint() const Q_DECL_OVERRIDE; - QLayoutItem *takeAt(int index) Q_DECL_OVERRIDE; - -private: - int doLayout(const QRect &rect, bool testOnly) const; - int smartSpacing(QStyle::PixelMetric pm) const; - - QList itemList; - int m_hSpace; - int m_vSpace; -}; - -#endif // FLOWLAYOUT_H From 7886fdcb69ec4db04a13fa5f5ef459342730fef9 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Wed, 19 Aug 2015 12:42:54 -0400 Subject: [PATCH 8/9] Video: resizable group avatar, synchronize avatar --- src/video/groupnetcamview.cpp | 119 +++++++++++++++++++--------- src/video/groupnetcamview.h | 2 + src/video/netcamview.cpp | 13 +++ src/video/netcamview.h | 1 + src/video/videosurface.cpp | 1 + src/widget/form/settings/avform.cpp | 8 ++ 6 files changed, 107 insertions(+), 37 deletions(-) diff --git a/src/video/groupnetcamview.cpp b/src/video/groupnetcamview.cpp index 06e2dbcce..3824e0eb8 100644 --- a/src/video/groupnetcamview.cpp +++ b/src/video/groupnetcamview.cpp @@ -20,27 +20,32 @@ #include "groupnetcamview.h" #include "src/widget/tool/croppinglabel.h" #include "src/video/videosurface.h" -#include -#include -#include +#include "src/persistence/settings.h" #include "src/audio/audio.h" #include "src/core/core.h" +#include "src/friendlist.h" +#include "src/friend.h" #include +#include +#include +#include +#include +#include class LabeledVideo : public QFrame { public: - LabeledVideo(QWidget* parent = 0, bool expanding = true) + LabeledVideo(const QPixmap& avatar, QWidget* parent = 0, bool expanding = true) : QFrame(parent) { //setFrameStyle(QFrame::Box); - videoSurface = new VideoSurface(QPixmap(), 0, expanding); + qDebug() << "Created expanding? " << expanding; + videoSurface = new VideoSurface(avatar, 0, expanding); videoSurface->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); - videoSurface->setMinimumHeight(96); + videoSurface->setMinimumHeight(32); //videoSurface->setMaximumHeight(96); connect(videoSurface, &VideoSurface::ratioChanged, this, &LabeledVideo::updateSize); label = new CroppingLabel(this); - label->setText("Unknown"); label->setTextFormat(Qt::PlainText); label->setStyleSheet("color: white"); //label->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); @@ -50,8 +55,6 @@ public: QVBoxLayout* layout = new QVBoxLayout(this); layout->addWidget(videoSurface, 1); layout->addWidget(label); - - setMouseTracking(true); } ~LabeledVideo() @@ -74,30 +77,33 @@ public: return label->text(); } + void setActive(bool active = true) + { + if (active) + setStyleSheet("QFrame { background-color: #414141; border-radius: 10px; }"); + else + setStyleSheet(QString()); + } + protected: void resizeEvent(QResizeEvent* event) final override { - QWidget::resizeEvent(event); + qDebug() << "Resize!"; updateSize(); - } - - void mousePressEvent(QMouseEvent* event) final override - { - if (videoSurface->isExpanding()) - { - setStyleSheet("QFrame { background-color: #414141; border-radius: 10px; }"); - selected = true; - } + QWidget::resizeEvent(event); } private slots: void updateSize() { + qDebug() << videoSurface->isExpanding(); if (videoSurface->isExpanding()) { int width = videoSurface->height() * videoSurface->getRatio(); - videoSurface->setFixedWidth(width); - setMaximumWidth(width + layout()->margin() * 2); + videoSurface->setMinimumWidth(width); + videoSurface->setMaximumWidth(width); + //setMaximumWidth(width + layout()->margin() * 2); + qDebug() << videoSurface->minimumWidth(); } } @@ -111,35 +117,40 @@ GroupNetCamView::GroupNetCamView(int group, QWidget *parent) : GenericNetCamView(parent) , group(group) { - videoLabelSurface = new LabeledVideo(this, false); + videoLabelSurface = new LabeledVideo(QPixmap(), this, false); videoSurface = videoLabelSurface->getVideoSurface(); //videoSurface->setExpanding(false); videoSurface->setMinimumHeight(256); - videoSurface->setContentsMargins(6, 6, 6, 6); + videoSurface->setContentsMargins(6, 6, 6, 0); videoLabelSurface->setContentsMargins(0, 0, 0, 0); videoLabelSurface->layout()->setMargin(0); videoLabelSurface->setStyleSheet("QFrame { background-color: black; }"); - verLayout->insertWidget(0, videoLabelSurface, 1); + //verLayout->insertWidget(0, videoLabelSurface, 1); + + QSplitter* splitter = new QSplitter(Qt::Vertical, this); + verLayout->insertWidget(0, splitter, 1); + splitter->addWidget(videoLabelSurface); + splitter->setStyleSheet("QSplitter { background-color: black; } QSplitter::handle { background-color: black; }"); QScrollArea* scrollArea = new QScrollArea(); scrollArea->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); - scrollArea->setStyleSheet("QScrollArea { background-color: black; }"); scrollArea->setFrameStyle(QFrame::NoFrame); QWidget* widget = new QWidget(nullptr); - scrollArea->setWidget(widget); scrollArea->setWidgetResizable(true); horLayout = new QHBoxLayout(widget); //FlowLayout* horLayout = new FlowLayout(widget); - horLayout->addStretch(); + horLayout->addStretch(1); - selfVideoSurface = new LabeledVideo(this); + selfVideoSurface = new LabeledVideo(Settings::getInstance().getSavedAvatar(Core::getInstance()->getSelfId().toString()), this); horLayout->addWidget(selfVideoSurface); - horLayout->setAlignment(selfVideoSurface, Qt::AlignCenter | Qt::AlignHCenter); + //horLayout->setAlignment(selfVideoSurface, Qt::AlignCenter | Qt::AlignHCenter); - horLayout->addStretch(); - verLayout->insertWidget(1, scrollArea); - scrollArea->setMinimumHeight(selfVideoSurface->height()); + horLayout->addStretch(1); + splitter->addWidget(scrollArea); + scrollArea->setWidget(widget); + //verLayout->insertWidget(1, scrollArea); + //scrollArea->setMinimumHeight(selfVideoSurface->height()); connect(&Audio::getInstance(), &Audio::groupAudioPlayed, this, &GroupNetCamView::groupAudioPlayed); @@ -148,9 +159,9 @@ GroupNetCamView::GroupNetCamView(int group, QWidget *parent) connect(timer, &QTimer::timeout, this, &GroupNetCamView::findActivePeer); timer->start(); - connect(Core::getInstance(), &Core::selfAvatarChanged, [this]() + connect(Core::getInstance(), &Core::selfAvatarChanged, [this](const QPixmap& pixmap) { - selfVideoSurface->update(); + selfVideoSurface->getVideoSurface()->setAvatar(pixmap); findActivePeer(); }); connect(Core::getInstance(), &Core::usernameSet, [this](const QString& username) @@ -158,6 +169,8 @@ GroupNetCamView::GroupNetCamView(int group, QWidget *parent) selfVideoSurface->setText(username); findActivePeer(); }); + connect(Core::getInstance(), &Core::friendAvatarChanged, this, &GroupNetCamView::friendAvatarChanged); + selfVideoSurface->setText(Core::getInstance()->getUsername()); } @@ -171,10 +184,10 @@ void GroupNetCamView::clearPeers() void GroupNetCamView::addPeer(int peer, const QString& name) { - LabeledVideo* labeledVideo = new LabeledVideo(this); + QPixmap groupAvatar = Settings::getInstance().getSavedAvatar(Core::getInstance()->getGroupPeerToxId(group, peer).toString()); + LabeledVideo* labeledVideo = new LabeledVideo(groupAvatar, this); labeledVideo->setText(name); horLayout->insertWidget(horLayout->count() - 1, labeledVideo); - horLayout->setAlignment(labeledVideo, Qt::AlignCenter | Qt::AlignHCenter); PeerVideo peerVideo; peerVideo.video = labeledVideo; videoList.insert(peer, peerVideo); @@ -202,6 +215,7 @@ void GroupNetCamView::setActive(int peer) if (peer == -1) { videoLabelSurface->setText(selfVideoSurface->getText()); + activePeer = -1; return; } @@ -212,7 +226,17 @@ void GroupNetCamView::setActive(int peer) // When group video exists: // videoSurface->setSource(peerVideo.value()->getVideoSurface()->source); - videoLabelSurface->setText(peerVideo.value().video->getText()); + auto lastVideo = videoList.find(activePeer); + + if (lastVideo != videoList.end()) + lastVideo.value().video->setActive(false); + + LabeledVideo *labeledVideo = peerVideo.value().video; + videoLabelSurface->setText(labeledVideo->getText()); + videoLabelSurface->getVideoSurface()->setAvatar(labeledVideo->getVideoSurface()->getAvatar()); + labeledVideo->setActive(); + + activePeer = peer; } } @@ -243,3 +267,24 @@ void GroupNetCamView::findActivePeer() setActive(candidate); } + +void GroupNetCamView::friendAvatarChanged(int FriendId, const QPixmap &pixmap) +{ + Friend* f = FriendList::findFriend(FriendId); + + for (int i = 0; i < Core::getInstance()->getGroupNumberPeers(group); ++i) + { + if (Core::getInstance()->getGroupPeerToxId(group, i) == f->getToxId()) + { + auto peerVideo = videoList.find(i); + + if (peerVideo != videoList.end()) + { + peerVideo.value().video->getVideoSurface()->setAvatar(pixmap); + findActivePeer(); + } + + break; + } + } +} diff --git a/src/video/groupnetcamview.h b/src/video/groupnetcamview.h index 0994f8c2c..9991bac7d 100644 --- a/src/video/groupnetcamview.h +++ b/src/video/groupnetcamview.h @@ -39,6 +39,8 @@ public slots: private slots: void findActivePeer(); + void friendAvatarChanged(int FriendId, const QPixmap& pixmap); + //void update private: struct PeerVideo diff --git a/src/video/netcamview.cpp b/src/video/netcamview.cpp index 5dbf83115..231ec1b3c 100644 --- a/src/video/netcamview.cpp +++ b/src/video/netcamview.cpp @@ -32,6 +32,7 @@ NetCamView::NetCamView(int friendId, QWidget* parent) : GenericNetCamView(parent) , selfFrame{nullptr} + , friendId{friendId} { QString id = FriendList::findFriend(friendId)->getToxId().toString(); videoSurface = new VideoSurface(Settings::getInstance().getSavedAvatar(id), this); @@ -70,10 +71,22 @@ NetCamView::NetCamView(int friendId, QWidget* parent) selfFrame->resetBoundary(boundingRect); }); + connect(Core::getInstance(), &Core::selfAvatarChanged, [this](const QPixmap& pixmap) + { + selfVideoSurface->setAvatar(pixmap); + }); + + connect(Core::getInstance(), &Core::friendAvatarChanged, [this](int FriendId, const QPixmap& pixmap) + { + if (this->friendId == FriendId) + videoSurface->setAvatar(pixmap); + }); + VideoMode videoMode; QSize videoSize = Settings::getInstance().getCamVideoRes(); videoMode.width = videoSize.width(); videoMode.height = videoSize.height(); + qDebug() << "SIZER" << videoSize; videoMode.FPS = Settings::getInstance().getCamVideoFPS(); CameraSource::getInstance().open(Settings::getInstance().getVideoDev(), videoMode); } diff --git a/src/video/netcamview.h b/src/video/netcamview.h index 0015b1755..2aa1309c7 100644 --- a/src/video/netcamview.h +++ b/src/video/netcamview.h @@ -52,6 +52,7 @@ private: VideoSurface* selfVideoSurface; MovableWidget* selfFrame; + int friendId; bool e = false; }; diff --git a/src/video/videosurface.cpp b/src/video/videosurface.cpp index 1402d3366..ae284b2d0 100644 --- a/src/video/videosurface.cpp +++ b/src/video/videosurface.cpp @@ -85,6 +85,7 @@ float VideoSurface::getRatio() const void VideoSurface::setAvatar(const QPixmap &pixmap) { avatar = pixmap; + update(); } QPixmap VideoSurface::getAvatar() const diff --git a/src/widget/form/settings/avform.cpp b/src/widget/form/settings/avform.cpp index 45e07374c..90b0af65d 100644 --- a/src/widget/form/settings/avform.cpp +++ b/src/widget/form/settings/avform.cpp @@ -178,6 +178,14 @@ void AVForm::updateVideoModes(int curIndex) break; } } + + if (videoModes.size()) + { + bodyUI->videoModescomboBox->setUpdatesEnabled(false); + bodyUI->videoModescomboBox->setCurrentIndex(-1); + bodyUI->videoModescomboBox->setUpdatesEnabled(true); + bodyUI->videoModescomboBox->setCurrentIndex(0); + } } } From 7f5a95a5994b49b197c1b7ec59247f50274de7c9 Mon Sep 17 00:00:00 2001 From: TheSpiritXIII Date: Wed, 19 Aug 2015 19:13:45 -0400 Subject: [PATCH 9/9] Video: Cleanup --- src/video/groupnetcamview.cpp | 1 + src/video/groupnetcamview.h | 1 - src/widget/tool/movablewidget.cpp | 54 ------------------------------- src/widget/tool/movablewidget.h | 1 - 4 files changed, 1 insertion(+), 56 deletions(-) diff --git a/src/video/groupnetcamview.cpp b/src/video/groupnetcamview.cpp index 3824e0eb8..3176f45a8 100644 --- a/src/video/groupnetcamview.cpp +++ b/src/video/groupnetcamview.cpp @@ -129,6 +129,7 @@ GroupNetCamView::GroupNetCamView(int group, QWidget *parent) //verLayout->insertWidget(0, videoLabelSurface, 1); QSplitter* splitter = new QSplitter(Qt::Vertical, this); + splitter->setChildrenCollapsible(false); verLayout->insertWidget(0, splitter, 1); splitter->addWidget(videoLabelSurface); splitter->setStyleSheet("QSplitter { background-color: black; } QSplitter::handle { background-color: black; }"); diff --git a/src/video/groupnetcamview.h b/src/video/groupnetcamview.h index 9991bac7d..aab80f34c 100644 --- a/src/video/groupnetcamview.h +++ b/src/video/groupnetcamview.h @@ -40,7 +40,6 @@ public slots: private slots: void findActivePeer(); void friendAvatarChanged(int FriendId, const QPixmap& pixmap); - //void update private: struct PeerVideo diff --git a/src/widget/tool/movablewidget.cpp b/src/widget/tool/movablewidget.cpp index 4fd770181..737e6350e 100644 --- a/src/widget/tool/movablewidget.cpp +++ b/src/widget/tool/movablewidget.cpp @@ -20,19 +20,10 @@ #include "movablewidget.h" #include #include -#include - -#include - -static float getNetcamX = 0.0f; -static float getNetcamY = 0.0f; -static float getNetcamW = 0.0f; -static float getNetcamH = 0.0f; MovableWidget::MovableWidget(QWidget *parent) : QWidget(parent) { - setMouseTracking(true); setMinimumHeight(64); setSizePolicy(QSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored)); actualSize = minimumSize(); @@ -40,53 +31,17 @@ MovableWidget::MovableWidget(QWidget *parent) setRatio(1.0f); resize(minimumSize()); actualPos = QPoint(0, 0); - - //move() } -/*void MovableWidget::setBoundary(QSize parentSize, QSize oldSize, float xPercent, float yPercent) -{ - // NOTE: When called, the parentWidget has not resized yet. - - // Prevent division with 0. - if (width() == oldSize.width() || height() == oldSize.height()) - return; - - float percentageX = x() / static_cast(oldSize.width() - width()); - float percentageY = y() / static_cast(oldSize.height() - height()); - - actualSize.setWidth(actualSize.width() * xPercent); - actualSize.setHeight(actualSize.height() * yPercent); - - if (actualSize.width() == 0) - actualSize.setWidth(1); - - if (actualSize.height() == 0) - actualSize.setHeight(1); - - resize(QSize(round(actualSize.width()), round(actualSize.height()))); - updateGeometry(); - - actualPos = QPointF(percentageX * (parentSize.width() - width()), percentageY * (parentSize.height() - height())); - - QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); - move(moveTo); -}*/ - void MovableWidget::resetBoundary(QRect newBoundary) { boundaryRect = newBoundary; - - //actualSize = QSize(getNetcamW * newBoundary.width(), getNetcamH * newBoundary.height()); resize(QSize(round(actualSize.width()), round(actualSize.height()))); - //actualPos = QPointF(getNetcamX * (newBoundary.width() - width()), getNetcamY * (newBoundary.height() - height())); QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); checkBoundary(moveTo); move(moveTo); actualPos = moveTo; - - qDebug() << "GGGG" << geometry(); } void MovableWidget::setBoundary(QRect newBoundary) @@ -120,7 +75,6 @@ void MovableWidget::setBoundary(QRect newBoundary) QPoint moveTo = QPoint(round(actualPos.x()), round(actualPos.y())); move(moveTo); - qDebug() << "hhhhh" << geometry() << boundaryRect << newBoundary; boundaryRect = newBoundary; } @@ -140,7 +94,6 @@ void MovableWidget::setRatio(float r) actualPos = pos(); actualSize = size(); - qDebug() << "GEO@" << geometry(); } void MovableWidget::mousePressEvent(QMouseEvent* event) @@ -313,13 +266,6 @@ void MovableWidget::mouseDoubleClickEvent(QMouseEvent* event) setGraphicsEffect(nullptr); } } -#include -void MovableWidget::paintEvent(QPaintEvent*) -{ - QPainter painter(this); - painter.setBrush(Qt::black); - painter.drawRect(rect()); -} void MovableWidget::checkBoundary(QPoint& point) const { diff --git a/src/widget/tool/movablewidget.h b/src/widget/tool/movablewidget.h index 86f1ef072..ffa7354c0 100644 --- a/src/widget/tool/movablewidget.h +++ b/src/widget/tool/movablewidget.h @@ -36,7 +36,6 @@ protected: void mouseMoveEvent(QMouseEvent* event); void mouseReleaseEvent(QMouseEvent* event); void mouseDoubleClickEvent(QMouseEvent* event); - void paintEvent(QPaintEvent*); private: void checkBoundary(QPoint& point) const;