mirror of https://github.com/procxx/kepka.git
First prototype of picture-in-picture player.
This commit is contained in:
parent
9feea4a724
commit
ca5c9271a3
|
@ -672,14 +672,16 @@ PRIVATE
|
||||||
media/streaming/media_streaming_utility.h
|
media/streaming/media_streaming_utility.h
|
||||||
media/streaming/media_streaming_video_track.cpp
|
media/streaming/media_streaming_video_track.cpp
|
||||||
media/streaming/media_streaming_video_track.h
|
media/streaming/media_streaming_video_track.h
|
||||||
media/view/media_view_playback_controls.cpp
|
|
||||||
media/view/media_view_playback_controls.h
|
|
||||||
media/view/media_view_playback_progress.cpp
|
|
||||||
media/view/media_view_playback_progress.h
|
|
||||||
media/view/media_view_group_thumbs.cpp
|
media/view/media_view_group_thumbs.cpp
|
||||||
media/view/media_view_group_thumbs.h
|
media/view/media_view_group_thumbs.h
|
||||||
media/view/media_view_overlay_widget.cpp
|
media/view/media_view_overlay_widget.cpp
|
||||||
media/view/media_view_overlay_widget.h
|
media/view/media_view_overlay_widget.h
|
||||||
|
media/view/media_view_pip.cpp
|
||||||
|
media/view/media_view_pip.h
|
||||||
|
media/view/media_view_playback_controls.cpp
|
||||||
|
media/view/media_view_playback_controls.h
|
||||||
|
media/view/media_view_playback_progress.cpp
|
||||||
|
media/view/media_view_playback_progress.h
|
||||||
mtproto/config_loader.cpp
|
mtproto/config_loader.cpp
|
||||||
mtproto/config_loader.h
|
mtproto/config_loader.h
|
||||||
mtproto/connection_abstract.cpp
|
mtproto/connection_abstract.cpp
|
||||||
|
|
|
@ -60,6 +60,10 @@ const Information &Document::info() const {
|
||||||
return _info;
|
return _info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
not_null<DocumentData*> Document::data() const {
|
||||||
|
return _document;
|
||||||
|
}
|
||||||
|
|
||||||
void Document::play(const PlaybackOptions &options) {
|
void Document::play(const PlaybackOptions &options) {
|
||||||
_player.play(options);
|
_player.play(options);
|
||||||
_info.audio.state.position
|
_info.audio.state.position
|
||||||
|
|
|
@ -31,6 +31,7 @@ public:
|
||||||
[[nodiscard]] Player &player();
|
[[nodiscard]] Player &player();
|
||||||
[[nodiscard]] const Player &player() const;
|
[[nodiscard]] const Player &player() const;
|
||||||
[[nodiscard]] const Information &info() const;
|
[[nodiscard]] const Information &info() const;
|
||||||
|
[[nodiscard]] not_null<DocumentData*> data() const;
|
||||||
|
|
||||||
[[nodiscard]] bool waitingShown() const;
|
[[nodiscard]] bool waitingShown() const;
|
||||||
[[nodiscard]] float64 waitingOpacity() const;
|
[[nodiscard]] float64 waitingOpacity() const;
|
||||||
|
|
|
@ -46,6 +46,10 @@ bool Instance::valid() const {
|
||||||
return (_shared != nullptr);
|
return (_shared != nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::shared_ptr<Document> Instance::shared() const {
|
||||||
|
return _shared;
|
||||||
|
}
|
||||||
|
|
||||||
const Player &Instance::player() const {
|
const Player &Instance::player() const {
|
||||||
Expects(_shared != nullptr);
|
Expects(_shared != nullptr);
|
||||||
|
|
||||||
|
|
|
@ -37,6 +37,7 @@ public:
|
||||||
~Instance();
|
~Instance();
|
||||||
|
|
||||||
[[nodiscard]] bool valid() const;
|
[[nodiscard]] bool valid() const;
|
||||||
|
[[nodiscard]] std::shared_ptr<Document> shared() const;
|
||||||
|
|
||||||
[[nodiscard]] const Player &player() const;
|
[[nodiscard]] const Player &player() const;
|
||||||
[[nodiscard]] const Information &info() const;
|
[[nodiscard]] const Information &info() const;
|
||||||
|
|
|
@ -26,6 +26,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "media/audio/media_audio.h"
|
#include "media/audio/media_audio.h"
|
||||||
#include "media/view/media_view_playback_controls.h"
|
#include "media/view/media_view_playback_controls.h"
|
||||||
#include "media/view/media_view_group_thumbs.h"
|
#include "media/view/media_view_group_thumbs.h"
|
||||||
|
#include "media/view/media_view_pip.h"
|
||||||
#include "media/streaming/media_streaming_instance.h"
|
#include "media/streaming/media_streaming_instance.h"
|
||||||
#include "media/streaming/media_streaming_player.h"
|
#include "media/streaming/media_streaming_player.h"
|
||||||
#include "media/player/media_player_instance.h"
|
#include "media/player/media_player_instance.h"
|
||||||
|
@ -186,13 +187,12 @@ struct OverlayWidget::Collage {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct OverlayWidget::Streamed {
|
struct OverlayWidget::Streamed {
|
||||||
template <typename Callback>
|
|
||||||
Streamed(
|
Streamed(
|
||||||
not_null<DocumentData*> document,
|
not_null<DocumentData*> document,
|
||||||
Data::FileOrigin origin,
|
Data::FileOrigin origin,
|
||||||
QWidget *controlsParent,
|
QWidget *controlsParent,
|
||||||
not_null<PlaybackControls::Delegate*> controlsDelegate,
|
not_null<PlaybackControls::Delegate*> controlsDelegate,
|
||||||
Callback &&loadingCallback);
|
Fn<void()> waitingCallback);
|
||||||
|
|
||||||
Streaming::Instance instance;
|
Streaming::Instance instance;
|
||||||
PlaybackControls controls;
|
PlaybackControls controls;
|
||||||
|
@ -204,14 +204,13 @@ struct OverlayWidget::Streamed {
|
||||||
bool resumeOnCallEnd = false;
|
bool resumeOnCallEnd = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Callback>
|
|
||||||
OverlayWidget::Streamed::Streamed(
|
OverlayWidget::Streamed::Streamed(
|
||||||
not_null<DocumentData*> document,
|
not_null<DocumentData*> document,
|
||||||
Data::FileOrigin origin,
|
Data::FileOrigin origin,
|
||||||
QWidget *controlsParent,
|
QWidget *controlsParent,
|
||||||
not_null<PlaybackControls::Delegate*> controlsDelegate,
|
not_null<PlaybackControls::Delegate*> controlsDelegate,
|
||||||
Callback &&loadingCallback)
|
Fn<void()> waitingCallback)
|
||||||
: instance(document, origin, std::forward<Callback>(loadingCallback))
|
: instance(document, origin, std::move(waitingCallback))
|
||||||
, controls(controlsParent, controlsDelegate) {
|
, controls(controlsParent, controlsDelegate) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1759,20 +1758,23 @@ void OverlayWidget::showPhoto(not_null<PhotoData*> photo, not_null<PeerData*> co
|
||||||
activateControls();
|
activateControls();
|
||||||
}
|
}
|
||||||
|
|
||||||
void OverlayWidget::showDocument(not_null<DocumentData*> document, HistoryItem *context) {
|
void OverlayWidget::showDocument(
|
||||||
showDocument(document, context, Data::CloudTheme());
|
not_null<DocumentData*> document,
|
||||||
|
HistoryItem *context) {
|
||||||
|
showDocument(document, context, Data::CloudTheme(), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void OverlayWidget::showTheme(
|
void OverlayWidget::showTheme(
|
||||||
not_null<DocumentData*> document,
|
not_null<DocumentData*> document,
|
||||||
const Data::CloudTheme &cloud) {
|
const Data::CloudTheme &cloud) {
|
||||||
showDocument(document, nullptr, cloud);
|
showDocument(document, nullptr, cloud, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void OverlayWidget::showDocument(
|
void OverlayWidget::showDocument(
|
||||||
not_null<DocumentData*> document,
|
not_null<DocumentData*> document,
|
||||||
HistoryItem *context,
|
HistoryItem *context,
|
||||||
const Data::CloudTheme &cloud) {
|
const Data::CloudTheme &cloud,
|
||||||
|
bool continueStreaming) {
|
||||||
if (context) {
|
if (context) {
|
||||||
setContext(context);
|
setContext(context);
|
||||||
} else {
|
} else {
|
||||||
|
@ -1783,7 +1785,7 @@ void OverlayWidget::showDocument(
|
||||||
_photo = nullptr;
|
_photo = nullptr;
|
||||||
|
|
||||||
_streamingStartPaused = false;
|
_streamingStartPaused = false;
|
||||||
displayDocument(document, context, cloud);
|
displayDocument(document, context, cloud, continueStreaming);
|
||||||
preloadData(0);
|
preloadData(0);
|
||||||
activateControls();
|
activateControls();
|
||||||
}
|
}
|
||||||
|
@ -1845,7 +1847,8 @@ void OverlayWidget::redisplayContent() {
|
||||||
void OverlayWidget::displayDocument(
|
void OverlayWidget::displayDocument(
|
||||||
DocumentData *doc,
|
DocumentData *doc,
|
||||||
HistoryItem *item,
|
HistoryItem *item,
|
||||||
const Data::CloudTheme &cloud) {
|
const Data::CloudTheme &cloud,
|
||||||
|
bool continueStreaming) {
|
||||||
if (isHidden()) {
|
if (isHidden()) {
|
||||||
moveToScreen();
|
moveToScreen();
|
||||||
}
|
}
|
||||||
|
@ -1870,7 +1873,7 @@ void OverlayWidget::displayDocument(
|
||||||
_doc->dimensions.height());
|
_doc->dimensions.height());
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (_doc->canBePlayed() && initStreaming()) {
|
if (_doc->canBePlayed() && initStreaming(continueStreaming)) {
|
||||||
} else if (_doc->isVideoFile()) {
|
} else if (_doc->isVideoFile()) {
|
||||||
_doc->automaticLoad(fileOrigin(), item);
|
_doc->automaticLoad(fileOrigin(), item);
|
||||||
initStreamingThumbnail();
|
initStreamingThumbnail();
|
||||||
|
@ -2001,7 +2004,7 @@ void OverlayWidget::displayFinished() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OverlayWidget::initStreaming() {
|
bool OverlayWidget::initStreaming(bool continueStreaming) {
|
||||||
Expects(_doc != nullptr);
|
Expects(_doc != nullptr);
|
||||||
Expects(_doc->canBePlayed());
|
Expects(_doc->canBePlayed());
|
||||||
|
|
||||||
|
@ -2023,16 +2026,29 @@ bool OverlayWidget::initStreaming() {
|
||||||
handleStreamingError(std::move(error));
|
handleStreamingError(std::move(error));
|
||||||
}, _streamed->instance.lifetime());
|
}, _streamed->instance.lifetime());
|
||||||
|
|
||||||
startStreamingPlayer();
|
if (continueStreaming) {
|
||||||
|
_pip = nullptr;
|
||||||
|
}
|
||||||
|
if (!continueStreaming
|
||||||
|
|| (!_streamed->instance.player().active()
|
||||||
|
&& !_streamed->instance.player().finished())) {
|
||||||
|
startStreamingPlayer();
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OverlayWidget::startStreamingPlayer() {
|
void OverlayWidget::startStreamingPlayer() {
|
||||||
Expects(_streamed != nullptr);
|
Expects(_streamed != nullptr);
|
||||||
|
|
||||||
if (!_streamed->withSound && _streamed->instance.player().playing()) {
|
if (_streamed->instance.player().playing()) {
|
||||||
|
if (!_streamed->withSound) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_pip = nullptr;
|
||||||
|
} else if (_pip && _streamed->withSound) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto position = _doc
|
const auto position = _doc
|
||||||
? _doc->session().settings().mediaLastPlaybackPosition(_doc->id)
|
? _doc->session().settings().mediaLastPlaybackPosition(_doc->id)
|
||||||
: 0;
|
: 0;
|
||||||
|
@ -2310,7 +2326,8 @@ void OverlayWidget::playbackPauseResume() {
|
||||||
if (!_doc->canBePlayed() || !initStreaming()) {
|
if (!_doc->canBePlayed() || !initStreaming()) {
|
||||||
redisplayContent();
|
redisplayContent();
|
||||||
}
|
}
|
||||||
} else if (_streamed->instance.player().finished()) {
|
} else if (_streamed->instance.player().finished()
|
||||||
|
|| !_streamed->instance.player().active()) {
|
||||||
_streamingStartPaused = false;
|
_streamingStartPaused = false;
|
||||||
restartAtSeekPosition(0);
|
restartAtSeekPosition(0);
|
||||||
} else if (_streamed->instance.player().paused()) {
|
} else if (_streamed->instance.player().paused()) {
|
||||||
|
@ -2338,6 +2355,8 @@ void OverlayWidget::restartAtSeekPosition(crl::time position) {
|
||||||
if (!_streamed->withSound) {
|
if (!_streamed->withSound) {
|
||||||
options.mode = Streaming::Mode::Video;
|
options.mode = Streaming::Mode::Video;
|
||||||
options.loop = true;
|
options.loop = true;
|
||||||
|
} else if (_pip) {
|
||||||
|
_pip = nullptr;
|
||||||
}
|
}
|
||||||
_streamed->instance.play(options);
|
_streamed->instance.play(options);
|
||||||
if (_streamingStartPaused) {
|
if (_streamingStartPaused) {
|
||||||
|
@ -2379,9 +2398,33 @@ float64 OverlayWidget::playbackControlsCurrentVolume() {
|
||||||
return Global::VideoVolume();
|
return Global::VideoVolume();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void OverlayWidget::switchToPip() {
|
||||||
|
const auto document = _doc;
|
||||||
|
const auto msgId = _msgid;
|
||||||
|
_pip = std::make_unique<Pip>(_streamed->instance.shared(), [=] {
|
||||||
|
showDocument(_doc, Auth().data().message(msgId), {}, true);
|
||||||
|
});
|
||||||
|
_pip->move(0, 0);
|
||||||
|
_pip->show();
|
||||||
|
_pip->events(
|
||||||
|
) | rpl::filter([=](not_null<QEvent*> e) {
|
||||||
|
return (e->type() == QEvent::Close);
|
||||||
|
}) | rpl::start_with_next([=] {
|
||||||
|
crl::on_main(_pip.get(), [=] {
|
||||||
|
_pip = nullptr;
|
||||||
|
});
|
||||||
|
}, _pip->lifetime());
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
|
||||||
void OverlayWidget::playbackToggleFullScreen() {
|
void OverlayWidget::playbackToggleFullScreen() {
|
||||||
Expects(_streamed != nullptr);
|
Expects(_streamed != nullptr);
|
||||||
|
|
||||||
|
if (!videoIsGifv() && !_fullScreenVideo) {
|
||||||
|
switchToPip();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!videoShown() || (videoIsGifv() && !_fullScreenVideo)) {
|
if (!videoShown() || (videoIsGifv() && !_fullScreenVideo)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -48,6 +48,7 @@ namespace Media {
|
||||||
namespace View {
|
namespace View {
|
||||||
|
|
||||||
class GroupThumbs;
|
class GroupThumbs;
|
||||||
|
class Pip;
|
||||||
|
|
||||||
#if defined Q_OS_MAC && !defined OS_MAC_OLD
|
#if defined Q_OS_MAC && !defined OS_MAC_OLD
|
||||||
#define USE_OPENGL_OVERLAY_WIDGET
|
#define USE_OPENGL_OVERLAY_WIDGET
|
||||||
|
@ -178,6 +179,7 @@ private:
|
||||||
void playbackPauseOnCall();
|
void playbackPauseOnCall();
|
||||||
void playbackResumeOnCall();
|
void playbackResumeOnCall();
|
||||||
void playbackPauseMusic();
|
void playbackPauseMusic();
|
||||||
|
void switchToPip();
|
||||||
|
|
||||||
void updateOver(QPoint mpos);
|
void updateOver(QPoint mpos);
|
||||||
void moveToScreen(bool force = false);
|
void moveToScreen(bool force = false);
|
||||||
|
@ -239,12 +241,14 @@ private:
|
||||||
void showDocument(
|
void showDocument(
|
||||||
not_null<DocumentData*> document,
|
not_null<DocumentData*> document,
|
||||||
HistoryItem *context,
|
HistoryItem *context,
|
||||||
const Data::CloudTheme &cloud);
|
const Data::CloudTheme &cloud,
|
||||||
|
bool continueStreaming);
|
||||||
void displayPhoto(not_null<PhotoData*> photo, HistoryItem *item);
|
void displayPhoto(not_null<PhotoData*> photo, HistoryItem *item);
|
||||||
void displayDocument(
|
void displayDocument(
|
||||||
DocumentData *document,
|
DocumentData *document,
|
||||||
HistoryItem *item,
|
HistoryItem *item,
|
||||||
const Data::CloudTheme &cloud = Data::CloudTheme());
|
const Data::CloudTheme &cloud = Data::CloudTheme(),
|
||||||
|
bool continueStreaming = false);
|
||||||
void displayFinished();
|
void displayFinished();
|
||||||
void redisplayContent();
|
void redisplayContent();
|
||||||
void findCurrent();
|
void findCurrent();
|
||||||
|
@ -258,7 +262,7 @@ private:
|
||||||
void refreshClipControllerGeometry();
|
void refreshClipControllerGeometry();
|
||||||
void refreshCaptionGeometry();
|
void refreshCaptionGeometry();
|
||||||
|
|
||||||
[[nodiscard]] bool initStreaming();
|
[[nodiscard]] bool initStreaming(bool continueStreaming = false);
|
||||||
void startStreamingPlayer();
|
void startStreamingPlayer();
|
||||||
void initStreamingThumbnail();
|
void initStreamingThumbnail();
|
||||||
void streamingReady(Streaming::Information &&info);
|
void streamingReady(Streaming::Information &&info);
|
||||||
|
@ -374,6 +378,7 @@ private:
|
||||||
bool _blurred = true;
|
bool _blurred = true;
|
||||||
|
|
||||||
std::unique_ptr<Streamed> _streamed;
|
std::unique_ptr<Streamed> _streamed;
|
||||||
|
std::unique_ptr<Pip> _pip;
|
||||||
|
|
||||||
const style::icon *_docIcon = nullptr;
|
const style::icon *_docIcon = nullptr;
|
||||||
style::color _docIconColor;
|
style::color _docIconColor;
|
||||||
|
|
|
@ -0,0 +1,264 @@
|
||||||
|
/*
|
||||||
|
This file is part of Telegram Desktop,
|
||||||
|
the official desktop application for the Telegram messaging service.
|
||||||
|
|
||||||
|
For license and copyright information please follow this link:
|
||||||
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
|
*/
|
||||||
|
#include "media/view/media_view_pip.h"
|
||||||
|
|
||||||
|
#include "media/streaming/media_streaming_player.h"
|
||||||
|
#include "media/streaming/media_streaming_document.h"
|
||||||
|
#include "core/application.h"
|
||||||
|
#include "window/window_controller.h"
|
||||||
|
#include "styles/style_window.h"
|
||||||
|
|
||||||
|
#include <QtGui/QWindow>
|
||||||
|
#include <QtGui/QScreen>
|
||||||
|
#include <QtWidgets/QApplication>
|
||||||
|
|
||||||
|
namespace Media {
|
||||||
|
namespace View {
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
constexpr auto kPipLoaderPriority = 2;
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
Pip::Pip(
|
||||||
|
std::shared_ptr<Streaming::Document> document,
|
||||||
|
FnMut<void()> closeAndContinue)
|
||||||
|
: _instance(document, [=] { waitingAnimationCallback(); })
|
||||||
|
, _closeAndContinue(std::move(closeAndContinue)) {
|
||||||
|
setWindowFlags(Qt::WindowStaysOnTopHint | Qt::FramelessWindowHint);
|
||||||
|
setupSize();
|
||||||
|
setupStreaming();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::setupSize() {
|
||||||
|
_size = style::ConvertScale(_instance.info().video.size);
|
||||||
|
if (_size.isEmpty()) {
|
||||||
|
_size = QSize(st::windowMinWidth, st::windowMinHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto widgetScreen = [&](auto &&widget) -> QScreen* {
|
||||||
|
if (auto handle = widget ? widget->windowHandle() : nullptr) {
|
||||||
|
return handle->screen();
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
};
|
||||||
|
const auto window = Core::App().activeWindow()
|
||||||
|
? Core::App().activeWindow()->widget().get()
|
||||||
|
: nullptr;
|
||||||
|
const auto activeWindowScreen = widgetScreen(window);
|
||||||
|
const auto myScreen = widgetScreen(this);
|
||||||
|
if (activeWindowScreen && myScreen && myScreen != activeWindowScreen) {
|
||||||
|
windowHandle()->setScreen(activeWindowScreen);
|
||||||
|
}
|
||||||
|
const auto screen = activeWindowScreen
|
||||||
|
? activeWindowScreen
|
||||||
|
: QGuiApplication::primaryScreen();
|
||||||
|
const auto available = screen->geometry();
|
||||||
|
const auto fit = QSize(available.width() / 2, available.height() / 2);
|
||||||
|
if (_size.width() > fit.width() || _size.height() > fit.height()) {
|
||||||
|
const auto byHeight = (fit.width() * _size.height() > fit.height() * _size.width());
|
||||||
|
_size = byHeight
|
||||||
|
? QSize(_size.width() * fit.height() / _size.height(), fit.height())
|
||||||
|
: QSize(fit.width(), _size.height() * fit.width() / _size.width());
|
||||||
|
}
|
||||||
|
resize(_size);
|
||||||
|
|
||||||
|
auto policy = QSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
|
||||||
|
policy.setHeightForWidth(true);
|
||||||
|
setSizePolicy(policy);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::setupStreaming() {
|
||||||
|
_instance.setPriority(kPipLoaderPriority);
|
||||||
|
_instance.lockPlayer();
|
||||||
|
|
||||||
|
_instance.player().updates(
|
||||||
|
) | rpl::start_with_next_error([=](Streaming::Update &&update) {
|
||||||
|
handleStreamingUpdate(std::move(update));
|
||||||
|
}, [=](Streaming::Error &&error) {
|
||||||
|
handleStreamingError(std::move(error));
|
||||||
|
}, _instance.lifetime());
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::handleStreamingUpdate(Streaming::Update &&update) {
|
||||||
|
using namespace Streaming;
|
||||||
|
|
||||||
|
update.data.match([&](Information &update) {
|
||||||
|
setupSize();
|
||||||
|
}, [&](const PreloadedVideo &update) {
|
||||||
|
//updatePlaybackState();
|
||||||
|
}, [&](const UpdateVideo &update) {
|
||||||
|
this->update();
|
||||||
|
Core::App().updateNonIdle();
|
||||||
|
//updatePlaybackState();
|
||||||
|
}, [&](const PreloadedAudio &update) {
|
||||||
|
//updatePlaybackState();
|
||||||
|
}, [&](const UpdateAudio &update) {
|
||||||
|
//updatePlaybackState();
|
||||||
|
}, [&](WaitingForData) {
|
||||||
|
}, [&](MutedByOther) {
|
||||||
|
}, [&](Finished) {
|
||||||
|
//updatePlaybackState();
|
||||||
|
close();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::handleStreamingError(Streaming::Error &&error) {
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::paintEvent(QPaintEvent *e) {
|
||||||
|
QPainter p(this);
|
||||||
|
|
||||||
|
const auto rect = QRect(QPoint(), size());
|
||||||
|
const auto image = videoFrameForDirectPaint();
|
||||||
|
const auto rotation = _instance.info().video.rotation;
|
||||||
|
const auto rotated = [](QRect rect, int rotation) {
|
||||||
|
switch (rotation) {
|
||||||
|
case 0: return rect;
|
||||||
|
case 90: return QRect(
|
||||||
|
rect.y(),
|
||||||
|
-rect.x() - rect.width(),
|
||||||
|
rect.height(),
|
||||||
|
rect.width());
|
||||||
|
case 180: return QRect(
|
||||||
|
-rect.x() - rect.width(),
|
||||||
|
-rect.y() - rect.height(),
|
||||||
|
rect.width(),
|
||||||
|
rect.height());
|
||||||
|
case 270: return QRect(
|
||||||
|
-rect.y() - rect.height(),
|
||||||
|
rect.x(),
|
||||||
|
rect.height(),
|
||||||
|
rect.width());
|
||||||
|
}
|
||||||
|
Unexpected("Rotation in OverlayWidget::paintTransformedVideoFrame");
|
||||||
|
};
|
||||||
|
|
||||||
|
PainterHighQualityEnabler hq(p);
|
||||||
|
if (rotation) {
|
||||||
|
p.save();
|
||||||
|
p.rotate(rotation);
|
||||||
|
}
|
||||||
|
p.drawImage(rotated(rect, rotation), image);
|
||||||
|
if (rotation) {
|
||||||
|
p.restore();
|
||||||
|
}
|
||||||
|
if (_instance.player().ready()) {
|
||||||
|
_instance.markFrameShown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::mousePressEvent(QMouseEvent *e) {
|
||||||
|
if (e->button() != Qt::LeftButton) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_pressPoint = e->globalPos();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::mouseReleaseEvent(QMouseEvent *e) {
|
||||||
|
if (e->button() != Qt::LeftButton || !base::take(_pressPoint)) {
|
||||||
|
return;
|
||||||
|
} else if (!base::take(_dragStartPosition)) {
|
||||||
|
playbackPauseResume();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::mouseMoveEvent(QMouseEvent *e) {
|
||||||
|
if (!_pressPoint) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto point = e->globalPos();
|
||||||
|
const auto distance = QApplication::startDragDistance();
|
||||||
|
if (!_dragStartPosition
|
||||||
|
&& (point - *_pressPoint).manhattanLength() > distance) {
|
||||||
|
_dragStartPosition = pos();
|
||||||
|
}
|
||||||
|
if (_dragStartPosition) {
|
||||||
|
updatePosition(point);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::keyPressEvent(QKeyEvent *e) {
|
||||||
|
if (e->key() == Qt::Key_Space) {
|
||||||
|
playbackPauseResume();
|
||||||
|
} else if (e->key() == Qt::Key_Escape) {
|
||||||
|
crl::on_main(this, [=] {
|
||||||
|
_closeAndContinue();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::playbackPauseResume() {
|
||||||
|
if (_instance.player().finished() || !_instance.player().active()) {
|
||||||
|
//restartAtSeekPosition(0);
|
||||||
|
} else if (_instance.player().paused()) {
|
||||||
|
_instance.resume();
|
||||||
|
//updatePlaybackState();
|
||||||
|
//playbackPauseMusic();
|
||||||
|
} else {
|
||||||
|
_instance.pause();
|
||||||
|
//updatePlaybackState();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::updatePosition(QPoint point) {
|
||||||
|
Expects(_dragStartPosition.has_value());
|
||||||
|
|
||||||
|
const auto position = *_dragStartPosition + (point - *_pressPoint);
|
||||||
|
move(position);
|
||||||
|
}
|
||||||
|
|
||||||
|
QImage Pip::videoFrame() const {
|
||||||
|
return _instance.player().ready()
|
||||||
|
? _instance.frame(Streaming::FrameRequest())
|
||||||
|
: _instance.info().video.cover;
|
||||||
|
}
|
||||||
|
|
||||||
|
QImage Pip::videoFrameForDirectPaint() const {
|
||||||
|
const auto result = videoFrame();
|
||||||
|
|
||||||
|
#ifdef USE_OPENGL_OVERLAY_WIDGET
|
||||||
|
const auto bytesPerLine = result.bytesPerLine();
|
||||||
|
if (bytesPerLine == result.width() * 4) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
// On macOS 10.8+ we use QOpenGLWidget as OverlayWidget base class.
|
||||||
|
// The OpenGL painter can't paint textures where byte data is with strides.
|
||||||
|
// So in that case we prepare a compact copy of the frame to render.
|
||||||
|
//
|
||||||
|
// See Qt commit ed557c037847e343caa010562952b398f806adcd
|
||||||
|
//
|
||||||
|
auto &cache = _streamed->frameForDirectPaint;
|
||||||
|
if (cache.size() != result.size()) {
|
||||||
|
cache = QImage(result.size(), result.format());
|
||||||
|
}
|
||||||
|
const auto height = result.height();
|
||||||
|
const auto line = cache.bytesPerLine();
|
||||||
|
Assert(line == result.width() * 4);
|
||||||
|
Assert(line < bytesPerLine);
|
||||||
|
|
||||||
|
auto from = result.bits();
|
||||||
|
auto to = cache.bits();
|
||||||
|
for (auto y = 0; y != height; ++y) {
|
||||||
|
memcpy(to, from, line);
|
||||||
|
to += line;
|
||||||
|
from += bytesPerLine;
|
||||||
|
}
|
||||||
|
return cache;
|
||||||
|
#endif // USE_OPENGL_OVERLAY_WIDGET
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Pip::waitingAnimationCallback() {
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace View
|
||||||
|
} // namespace Media
|
|
@ -0,0 +1,60 @@
|
||||||
|
/*
|
||||||
|
This file is part of Telegram Desktop,
|
||||||
|
the official desktop application for the Telegram messaging service.
|
||||||
|
|
||||||
|
For license and copyright information please follow this link:
|
||||||
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "media/streaming/media_streaming_instance.h"
|
||||||
|
#include "ui/rp_widget.h"
|
||||||
|
|
||||||
|
namespace Media {
|
||||||
|
namespace View {
|
||||||
|
|
||||||
|
#if defined Q_OS_MAC && !defined OS_MAC_OLD
|
||||||
|
#define USE_OPENGL_OVERLAY_WIDGET
|
||||||
|
#endif // Q_OS_MAC && !OS_MAC_OLD
|
||||||
|
|
||||||
|
#ifdef USE_OPENGL_OVERLAY_WIDGET
|
||||||
|
using PipParent = Ui::RpWidgetWrap<QOpenGLWidget>;
|
||||||
|
#else // USE_OPENGL_OVERLAY_WIDGET
|
||||||
|
using PipParent = Ui::RpWidget;
|
||||||
|
#endif // USE_OPENGL_OVERLAY_WIDGET
|
||||||
|
|
||||||
|
class Pip final : public PipParent {
|
||||||
|
public:
|
||||||
|
Pip(
|
||||||
|
std::shared_ptr<Streaming::Document> document,
|
||||||
|
FnMut<void()> closeAndContinue);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void paintEvent(QPaintEvent *e) override;
|
||||||
|
void mousePressEvent(QMouseEvent *e) override;
|
||||||
|
void mouseReleaseEvent(QMouseEvent *e) override;
|
||||||
|
void mouseMoveEvent(QMouseEvent *e) override;
|
||||||
|
void keyPressEvent(QKeyEvent *e) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void setupSize();
|
||||||
|
void setupStreaming();
|
||||||
|
void playbackPauseResume();
|
||||||
|
void updatePosition(QPoint point);
|
||||||
|
void waitingAnimationCallback();
|
||||||
|
void handleStreamingUpdate(Streaming::Update &&update);
|
||||||
|
void handleStreamingError(Streaming::Error &&error);
|
||||||
|
|
||||||
|
[[nodiscard]] QImage videoFrame() const;
|
||||||
|
[[nodiscard]] QImage videoFrameForDirectPaint() const;
|
||||||
|
|
||||||
|
Streaming::Instance _instance;
|
||||||
|
QSize _size;
|
||||||
|
std::optional<QPoint> _pressPoint;
|
||||||
|
std::optional<QPoint> _dragStartPosition;
|
||||||
|
FnMut<void()> _closeAndContinue;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace View
|
||||||
|
} // namespace Media
|
|
@ -207,3 +207,7 @@ themePreviewCancelButton: RoundButton(defaultLightButton) {
|
||||||
}
|
}
|
||||||
themePreviewButtonsSkip: 20px;
|
themePreviewButtonsSkip: 20px;
|
||||||
themePreviewDialogsWidth: 312px;
|
themePreviewDialogsWidth: 312px;
|
||||||
|
|
||||||
|
pipBorderSkip: 20px;
|
||||||
|
pipBorderSnapArea: 10px;
|
||||||
|
pip
|
Loading…
Reference in New Issue