Save video PiP window geometry.

This commit is contained in:
John Preston 2020-01-29 11:58:07 +03:00
parent 58dd33d8a2
commit f81f37505b
6 changed files with 193 additions and 47 deletions

View File

@ -44,12 +44,14 @@ Settings::Variables::Variables()
QByteArray Settings::serialize() const {
const auto autoDownload = _variables.autoDownload.serialize();
auto size = sizeof(qint32) * 30;
auto size = sizeof(qint32) * 38;
for (auto i = _variables.soundOverrides.cbegin(), e = _variables.soundOverrides.cend(); i != e; ++i) {
size += Serialize::stringSize(i.key()) + Serialize::stringSize(i.value());
}
size += _variables.groupStickersSectionHidden.size() * sizeof(quint64);
size += _variables.mediaLastPlaybackPosition.size() * 2 * sizeof(quint64);
size += Serialize::bytearraySize(autoDownload);
size += Serialize::bytearraySize(_variables.videoPipGeometry);
auto result = QByteArray();
result.reserve(size);
@ -107,6 +109,7 @@ QByteArray Settings::serialize() const {
stream << quint64(id) << qint64(time);
}
stream << qint32(SerializePlaybackSpeed(_variables.videoPlaybackSpeed.current()));
stream << _variables.videoPipGeometry;
}
return result;
}
@ -158,6 +161,7 @@ void Settings::constructFromSerialized(const QByteArray &serialized) {
qint32 spellcheckerEnabled = _variables.spellcheckerEnabled.current() ? 1 : 0;
std::vector<std::pair<DocumentId, crl::time>> mediaLastPlaybackPosition;
qint32 videoPlaybackSpeed = SerializePlaybackSpeed(_variables.videoPlaybackSpeed.current());
QByteArray videoPipGeometry = _variables.videoPipGeometry;
stream >> versionTag;
if (versionTag == kVersionTag) {
@ -281,6 +285,9 @@ void Settings::constructFromSerialized(const QByteArray &serialized) {
if (!stream.atEnd()) {
stream >> videoPlaybackSpeed;
}
if (!stream.atEnd()) {
stream >> videoPipGeometry;
}
if (stream.status() != QDataStream::Ok) {
LOG(("App Error: "
"Bad data for Main::Settings::constructFromSerialized()"));
@ -369,6 +376,7 @@ void Settings::constructFromSerialized(const QByteArray &serialized) {
_variables.spellcheckerEnabled = (spellcheckerEnabled == 1);
_variables.mediaLastPlaybackPosition = std::move(mediaLastPlaybackPosition);
_variables.videoPlaybackSpeed = DeserializePlaybackSpeed(videoPlaybackSpeed);
_variables.videoPipGeometry = videoPipGeometry;
}
void Settings::setSupportChatsTimeSlice(int slice) {

View File

@ -247,6 +247,12 @@ public:
void setVideoPlaybackSpeed(float64 speed) {
_variables.videoPlaybackSpeed = speed;
}
[[nodiscard]] QByteArray videoPipGeometry() const {
return _variables.videoPipGeometry;
}
void setVideoPipGeometry(QByteArray geometry) {
_variables.videoPipGeometry = geometry;
}
private:
struct Variables {
@ -289,6 +295,7 @@ private:
rpl::variable<bool> spellcheckerEnabled = true;
std::vector<std::pair<DocumentId, crl::time>> mediaLastPlaybackPosition;
rpl::variable<float64> videoPlaybackSpeed = 1.;
QByteArray videoPipGeometry;
static constexpr auto kDefaultSupportChatsLimitSlice
= 7 * 24 * 60 * 60;

View File

@ -83,6 +83,43 @@ constexpr auto kIdsLimit = 48;
// Preload next messages if we went further from current than that.
constexpr auto kIdsPreloadAfter = 28;
class PipDelegate final : public Pip::Delegate {
public:
PipDelegate(QWidget *parent, not_null<Main::Session*> session);
void pipSaveGeometry(QByteArray geometry) override;
QByteArray pipLoadGeometry() override;
float64 pipPlaybackSpeed() override;
QWidget *pipParentWidget() override;
private:
QWidget *_parent = nullptr;
not_null<Main::Session*> _session;
};
PipDelegate::PipDelegate(QWidget *parent, not_null<Main::Session*> session)
: _parent(parent)
, _session(session) {
}
void PipDelegate::pipSaveGeometry(QByteArray geometry) {
_session->settings().setVideoPipGeometry(geometry);
_session->saveSettingsDelayed();
}
QByteArray PipDelegate::pipLoadGeometry() {
return _session->settings().videoPipGeometry();
}
float64 PipDelegate::pipPlaybackSpeed() {
return _session->settings().videoPlaybackSpeed();
}
QWidget *PipDelegate::pipParentWidget() {
return _parent;
}
Images::Options VideoThumbOptions(not_null<DocumentData*> document) {
const auto result = Images::Option::Smooth | Images::Option::Blurred;
return (document && document->isVideoMessage())
@ -204,6 +241,21 @@ struct OverlayWidget::Streamed {
bool resumeOnCallEnd = false;
};
struct OverlayWidget::PipWrap {
PipWrap(
QWidget *parent,
not_null<DocumentData*> document,
std::shared_ptr<Streaming::Document> shared,
FnMut<void()> closeAndContinue,
FnMut<void()> destroy);
PipWrap(const PipWrap &other) = delete;
PipWrap &operator=(const PipWrap &other) = delete;
PipDelegate delegate;
Pip wrapped;
};
OverlayWidget::Streamed::Streamed(
not_null<DocumentData*> document,
Data::FileOrigin origin,
@ -214,6 +266,20 @@ OverlayWidget::Streamed::Streamed(
, controls(controlsParent, controlsDelegate) {
}
OverlayWidget::PipWrap::PipWrap(
QWidget *parent,
not_null<DocumentData*> document,
std::shared_ptr<Streaming::Document> shared,
FnMut<void()> closeAndContinue,
FnMut<void()> destroy)
: delegate(parent, &document->session())
, wrapped(
&delegate,
std::move(shared),
std::move(closeAndContinue),
std::move(destroy)) {
}
OverlayWidget::OverlayWidget()
: OverlayParent(nullptr)
, _transparentBrush(style::transparentPlaceholderBrush())
@ -985,6 +1051,7 @@ void OverlayWidget::clearData() {
_fromName = QString();
_photo = nullptr;
_doc = nullptr;
_pip = nullptr;
_fullScreenVideo = false;
_caption.clear();
}
@ -2440,13 +2507,20 @@ float64 OverlayWidget::playbackControlsCurrentSpeed() {
}
void OverlayWidget::switchToPip() {
Expects(_streamed != nullptr);
Expects(_doc != nullptr);
const auto document = _doc;
const auto msgId = _msgid;
_pip = std::make_unique<Pip>(this, _streamed->instance.shared(), [=] {
showDocument(_doc, Auth().data().message(msgId), {}, true);
}, [=] {
_pip = nullptr;
});
const auto closeAndContinue = [=] {
showDocument(document, document->owner().message(msgId), {}, true);
};
_pip = std::make_unique<PipWrap>(
this,
document,
_streamed->instance.shared(),
closeAndContinue,
[=] { _pip = nullptr; });
close();
}

View File

@ -129,6 +129,7 @@ private slots:
private:
struct Streamed;
struct PipWrap;
enum OverState {
OverNone,
@ -384,7 +385,7 @@ private:
bool _blurred = true;
std::unique_ptr<Streamed> _streamed;
std::unique_ptr<Pip> _pip;
std::unique_ptr<PipWrap> _pip;
const style::icon *_docIcon = nullptr;
style::color _docIconColor;

View File

@ -11,8 +11,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "media/streaming/media_streaming_document.h"
#include "media/streaming/media_streaming_utility.h"
#include "media/audio/media_audio.h"
#include "main/main_session.h"
#include "main/main_settings.h"
#include "data/data_document.h"
#include "core/application.h"
#include "ui/platform/ui_platform_utility.h"
@ -32,6 +30,7 @@ namespace View {
namespace {
constexpr auto kPipLoaderPriority = 2;
constexpr auto kSaveGeometryTimeout = crl::time(1000);
[[nodiscard]] QRect ScreenFromPosition(QPoint point) {
const auto screen = QGuiApplication::screenAt(point);
@ -199,6 +198,34 @@ constexpr auto kPipLoaderPriority = 2;
Unexpected("RectPart in PiP Constrained.");
}
[[nodiscard]] QByteArray Serialize(const PipPanel::Position &position) {
auto result = QByteArray();
auto stream = QDataStream(&result, QIODevice::WriteOnly);
stream.setVersion(QDataStream::Qt_5_3);
stream
<< qint32(position.attached.value())
<< qint32(position.snapped.value())
<< position.screen
<< position.geometry;
stream.device()->close();
return result;
}
[[nodiscard]] PipPanel::Position Deserialize(const QByteArray &data) {
auto stream = QDataStream(data);
auto result = PipPanel::Position();
auto attached = qint32(0);
auto snapped = qint32(0);
stream >> attached >> snapped >> result.screen >> result.geometry;
if (stream.status() != QDataStream::Ok) {
return {};
}
result.attached = RectParts::from_raw(attached);
result.snapped = RectParts::from_raw(snapped);
return result;
}
} // namespace
PipPanel::PipPanel(
@ -574,14 +601,16 @@ void PipPanel::moveAnimated(QPoint to) {
}
Pip::Pip(
QWidget *parent,
not_null<Delegate*> delegate,
std::shared_ptr<Streaming::Document> document,
FnMut<void()> closeAndContinue,
FnMut<void()> destroy)
: _instance(document, [=] { waitingAnimationCallback(); })
, _panel(parent, [=](QPainter &p, const FrameRequest &request) {
paint(p, request);
})
: _delegate(delegate)
, _instance(document, [=] { waitingAnimationCallback(); })
, _panel(
_delegate->pipParentWidget(),
[=](QPainter &p, const FrameRequest &request) { paint(p, request); })
, _saveGeometryTimer([=] { saveGeometry(); })
, _playPauseResume(
std::in_place,
&_panel,
@ -596,6 +625,46 @@ Pip::Pip(
object_ptr<Ui::IconButton>(&_panel, st::boxTitleClose))
, _closeAndContinue(std::move(closeAndContinue))
, _destroy(std::move(destroy)) {
setupPanel();
setupButtons();
setupStreaming();
}
void Pip::setupPanel() {
const auto size = style::ConvertScale(_instance.info().video.size);
if (size.isEmpty()) {
_panel.setAspectRatio(QSize(1, 1));
} else {
_panel.setAspectRatio(size);
}
_panel.setPosition(Deserialize(_delegate->pipLoadGeometry()));
_panel.show();
_panel.geometryValue(
) | rpl::skip(1) | rpl::start_with_next([=] {
_saveGeometryTimer.callOnce(kSaveGeometryTimeout);
}, _panel.lifetime());
_panel.events(
) | rpl::filter([=](not_null<QEvent*> e) {
return e->type() == QEvent::Close;
}) | rpl::start_with_next([=] {
_destroy();
}, _panel.lifetime());
}
void Pip::setupButtons() {
_panel.sizeValue(
) | rpl::start_with_next([=](QSize size) {
_close->moveToLeft(0, 0, size.width());
const auto skip = st::mediaviewFullScreenLeft;
const auto sum = _playPauseResume->width() + skip + _pictureInPicture->width();
const auto left = (size.width() - sum) / 2;
const auto top = size.height() - _playPauseResume->height() - skip;
_playPauseResume->moveToLeft(left, top);
_pictureInPicture->moveToRight(left, top);
}, _panel.lifetime());
_close->entity()->addClickHandler([=] {
_panel.close();
});
@ -608,37 +677,10 @@ Pip::Pip(
_close->show(anim::type::instant);
_pictureInPicture->show(anim::type::instant);
_playPauseResume->show(anim::type::instant);
setupPanel();
setupStreaming();
}
void Pip::setupPanel() {
const auto size = style::ConvertScale(_instance.info().video.size);
if (size.isEmpty()) {
_panel.setAspectRatio(QSize(1, 1));
} else {
_panel.setAspectRatio(size);
}
_panel.setPosition(PipPanel::Position());
_panel.show();
_panel.sizeValue(
) | rpl::start_with_next([=](QSize size) {
_close->moveToLeft(0, 0, size.width());
const auto skip = st::mediaviewFullScreenLeft;
const auto sum = _playPauseResume->width() + skip + _pictureInPicture->width();
const auto left = (size.width() - sum) / 2;
const auto top = size.height() - _playPauseResume->height() - skip;
_playPauseResume->moveToLeft(left, top);
_pictureInPicture->moveToRight(left, top);
}, _panel.lifetime());
_panel.events(
) | rpl::filter([=](not_null<QEvent*> e) {
return e->type() == QEvent::Close;
}) | rpl::start_with_next([=] {
_destroy();
}, _panel.lifetime());
void Pip::saveGeometry() {
_delegate->pipSaveGeometry(Serialize(_panel.countPosition()));
}
void Pip::updatePlayPauseResumeState(const Player::TrackState &state) {
@ -726,9 +768,7 @@ void Pip::restartAtSeekPosition(crl::time position) {
auto options = Streaming::PlaybackOptions();
options.position = position;
options.audioId = _instance.player().prepareLegacyState().id;
options.speed = options.audioId.audio()
? options.audioId.audio()->session().settings().videoPlaybackSpeed()
: 1.;
options.speed = _delegate->pipPlaybackSpeed();
_instance.play(options);
updatePlaybackState();
}

View File

@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "media/streaming/media_streaming_instance.h"
#include "ui/effects/animations.h"
#include "ui/rp_widget.h"
#include "base/timer.h"
#include <QtCore/QPointer>
@ -76,6 +77,9 @@ private:
RectParts _attached = RectParts();
QSize _ratio;
bool _useTransparency = true;
style::margins _padding;
RectPart _overState = RectPart();
std::optional<RectPart> _pressState;
QPoint _pressPoint;
@ -89,8 +93,16 @@ private:
class Pip final {
public:
class Delegate {
public:
virtual void pipSaveGeometry(QByteArray geometry) = 0;
[[nodiscard]] virtual QByteArray pipLoadGeometry() = 0;
[[nodiscard]] virtual float64 pipPlaybackSpeed() = 0;
[[nodiscard]] virtual QWidget *pipParentWidget() = 0;
};
Pip(
QWidget *parent,
not_null<Delegate*> delegate,
std::shared_ptr<Streaming::Document> document,
FnMut<void()> closeAndContinue,
FnMut<void()> destroy);
@ -99,12 +111,14 @@ private:
using FrameRequest = Streaming::FrameRequest;
void setupPanel();
void setupButtons();
void setupStreaming();
void paint(QPainter &p, FrameRequest request);
void playbackPauseResume();
void waitingAnimationCallback();
void handleStreamingUpdate(Streaming::Update &&update);
void handleStreamingError(Streaming::Error &&error);
void saveGeometry();
void updatePlaybackState();
void updatePlayPauseResumeState(const Player::TrackState &state);
@ -114,9 +128,11 @@ private:
[[nodiscard]] QImage videoFrameForDirectPaint(
const FrameRequest &request) const;
const not_null<Delegate*> _delegate;
Streaming::Instance _instance;
PipPanel _panel;
QSize _size;
base::Timer _saveGeometryTimer;
base::unique_qptr<Ui::FadeWrap<Ui::IconButton>> _playPauseResume;
base::unique_qptr< Ui::FadeWrap<Ui::IconButton>> _pictureInPicture;