Allow rotating content in media viewer.
|
@ -337,6 +337,8 @@ PRIVATE
|
||||||
data/data_groups.h
|
data/data_groups.h
|
||||||
data/data_location.cpp
|
data/data_location.cpp
|
||||||
data/data_location.h
|
data/data_location.h
|
||||||
|
data/data_media_rotation.cpp
|
||||||
|
data/data_media_rotation.h
|
||||||
data/data_media_types.cpp
|
data/data_media_types.cpp
|
||||||
data/data_media_types.h
|
data/data_media_types.h
|
||||||
data/data_messages.cpp
|
data/data_messages.cpp
|
||||||
|
|
Before Width: | Height: | Size: 177 B After Width: | Height: | Size: 513 B |
Before Width: | Height: | Size: 278 B After Width: | Height: | Size: 1.0 KiB |
Before Width: | Height: | Size: 417 B After Width: | Height: | Size: 1.6 KiB |
After Width: | Height: | Size: 912 B |
After Width: | Height: | Size: 1.8 KiB |
After Width: | Height: | Size: 2.5 KiB |
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
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 "data/data_media_rotation.h"
|
||||||
|
|
||||||
|
namespace Data {
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
[[nodiscard]] int NormalizeRotation(int rotation) {
|
||||||
|
const auto result = rotation
|
||||||
|
- ((rotation / 360) - ((rotation < 0) ? 1 : 0)) * 360;
|
||||||
|
|
||||||
|
Ensures(result >= 0 && result < 360);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
void MediaRotation::set(not_null<PhotoData*> photo, int rotation) {
|
||||||
|
if (rotation % 360) {
|
||||||
|
_photoRotations[photo] = NormalizeRotation(rotation);
|
||||||
|
} else {
|
||||||
|
_photoRotations.remove(photo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int MediaRotation::get(not_null<PhotoData*> photo) const {
|
||||||
|
const auto i = _photoRotations.find(photo);
|
||||||
|
return (i != end(_photoRotations)) ? i->second : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void MediaRotation::set(not_null<DocumentData*> document, int rotation) {
|
||||||
|
if (rotation % 360) {
|
||||||
|
_documentRotations[document] = NormalizeRotation(rotation);
|
||||||
|
} else {
|
||||||
|
_documentRotations.remove(document);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int MediaRotation::get(not_null<DocumentData*> document) const {
|
||||||
|
const auto i = _documentRotations.find(document);
|
||||||
|
return (i != end(_documentRotations)) ? i->second : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Data
|
|
@ -0,0 +1,29 @@
|
||||||
|
/*
|
||||||
|
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
|
||||||
|
|
||||||
|
class PhotoData;
|
||||||
|
class DocumentData;
|
||||||
|
|
||||||
|
namespace Data {
|
||||||
|
|
||||||
|
class MediaRotation final {
|
||||||
|
public:
|
||||||
|
void set(not_null<PhotoData*> photo, int rotation);
|
||||||
|
[[nodiscard]] int get(not_null<PhotoData*> photo) const;
|
||||||
|
|
||||||
|
void set(not_null<DocumentData*> document, int rotation);
|
||||||
|
[[nodiscard]] int get(not_null<DocumentData*> document) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
base::flat_map<not_null<PhotoData*>, int> _photoRotations;
|
||||||
|
base::flat_map<not_null<DocumentData*>, int> _documentRotations;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Data
|
|
@ -47,6 +47,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "data/data_scheduled_messages.h"
|
#include "data/data_scheduled_messages.h"
|
||||||
#include "data/data_cloud_themes.h"
|
#include "data/data_cloud_themes.h"
|
||||||
#include "data/data_streaming.h"
|
#include "data/data_streaming.h"
|
||||||
|
#include "data/data_media_rotation.h"
|
||||||
#include "base/platform/base_platform_info.h"
|
#include "base/platform/base_platform_info.h"
|
||||||
#include "base/unixtime.h"
|
#include "base/unixtime.h"
|
||||||
#include "base/call_delayed.h"
|
#include "base/call_delayed.h"
|
||||||
|
@ -192,7 +193,8 @@ Session::Session(not_null<Main::Session*> session)
|
||||||
, _groups(this)
|
, _groups(this)
|
||||||
, _scheduledMessages(std::make_unique<ScheduledMessages>(this))
|
, _scheduledMessages(std::make_unique<ScheduledMessages>(this))
|
||||||
, _cloudThemes(std::make_unique<CloudThemes>(session))
|
, _cloudThemes(std::make_unique<CloudThemes>(session))
|
||||||
, _streaming(std::make_unique<Streaming>(this)) {
|
, _streaming(std::make_unique<Streaming>(this))
|
||||||
|
, _mediaRotation(std::make_unique<MediaRotation>()) {
|
||||||
_cache->open(Local::cacheKey());
|
_cache->open(Local::cacheKey());
|
||||||
_bigFileCache->open(Local::cacheBigFileKey());
|
_bigFileCache->open(Local::cacheBigFileKey());
|
||||||
|
|
||||||
|
|
|
@ -60,6 +60,7 @@ class WallPaper;
|
||||||
class ScheduledMessages;
|
class ScheduledMessages;
|
||||||
class CloudThemes;
|
class CloudThemes;
|
||||||
class Streaming;
|
class Streaming;
|
||||||
|
class MediaRotation;
|
||||||
|
|
||||||
class Session final {
|
class Session final {
|
||||||
public:
|
public:
|
||||||
|
@ -92,6 +93,9 @@ public:
|
||||||
[[nodiscard]] Streaming &streaming() const {
|
[[nodiscard]] Streaming &streaming() const {
|
||||||
return *_streaming;
|
return *_streaming;
|
||||||
}
|
}
|
||||||
|
[[nodiscard]] MediaRotation &mediaRotation() const {
|
||||||
|
return *_mediaRotation;
|
||||||
|
}
|
||||||
[[nodiscard]] MsgId nextNonHistoryEntryId() {
|
[[nodiscard]] MsgId nextNonHistoryEntryId() {
|
||||||
return ++_nonHistoryEntryId;
|
return ++_nonHistoryEntryId;
|
||||||
}
|
}
|
||||||
|
@ -984,6 +988,7 @@ private:
|
||||||
std::unique_ptr<ScheduledMessages> _scheduledMessages;
|
std::unique_ptr<ScheduledMessages> _scheduledMessages;
|
||||||
std::unique_ptr<CloudThemes> _cloudThemes;
|
std::unique_ptr<CloudThemes> _cloudThemes;
|
||||||
std::unique_ptr<Streaming> _streaming;
|
std::unique_ptr<Streaming> _streaming;
|
||||||
|
std::unique_ptr<MediaRotation> _mediaRotation;
|
||||||
MsgId _nonHistoryEntryId = ServerMaxMsgId;
|
MsgId _nonHistoryEntryId = ServerMaxMsgId;
|
||||||
|
|
||||||
rpl::lifetime _lifetime;
|
rpl::lifetime _lifetime;
|
||||||
|
|
|
@ -104,6 +104,7 @@ mediaviewLeft: icon {{ "mediaview_next-flip_horizontal", mediaviewControlFg }};
|
||||||
mediaviewRight: icon {{ "mediaview_next", mediaviewControlFg }};
|
mediaviewRight: icon {{ "mediaview_next", mediaviewControlFg }};
|
||||||
mediaviewClose: icon {{ "mediaview_close", mediaviewControlFg }};
|
mediaviewClose: icon {{ "mediaview_close", mediaviewControlFg }};
|
||||||
mediaviewSave: icon {{ "mediaview_download", mediaviewControlFg }};
|
mediaviewSave: icon {{ "mediaview_download", mediaviewControlFg }};
|
||||||
|
mediaviewRotate: icon {{ "mediaview_rotate", mediaviewControlFg }};
|
||||||
mediaviewMore: icon {{ "mediaview_more", mediaviewControlFg }};
|
mediaviewMore: icon {{ "mediaview_more", mediaviewControlFg }};
|
||||||
|
|
||||||
mediaviewFileRed: icon {
|
mediaviewFileRed: icon {
|
||||||
|
|
|
@ -39,6 +39,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "data/data_chat.h"
|
#include "data/data_chat.h"
|
||||||
#include "data/data_user.h"
|
#include "data/data_user.h"
|
||||||
#include "data/data_file_origin.h"
|
#include "data/data_file_origin.h"
|
||||||
|
#include "data/data_media_rotation.h"
|
||||||
#include "window/themes/window_theme_preview.h"
|
#include "window/themes/window_theme_preview.h"
|
||||||
#include "window/window_peer_menu.h"
|
#include "window/window_peer_menu.h"
|
||||||
#include "window/window_session_controller.h"
|
#include "window/window_session_controller.h"
|
||||||
|
@ -198,6 +199,32 @@ QPixmap PrepareStaticImage(const QString &path) {
|
||||||
return App::pixmapFromImageInPlace(std::move(image));
|
return App::pixmapFromImageInPlace(std::move(image));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] QRect RotatedRect(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 RotatedRect.");
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] bool UsePainterRotation(int rotation) {
|
||||||
|
return Platform::IsMac() || !(rotation % 180);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
struct OverlayWidget::SharedMedia {
|
struct OverlayWidget::SharedMedia {
|
||||||
|
@ -432,6 +459,12 @@ void OverlayWidget::moveToScreen(bool force) {
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QSize OverlayWidget::flipSizeByRotation(QSize size) const {
|
||||||
|
return (((_rotation / 90) % 2) == 1)
|
||||||
|
? QSize(size.height(), size.width())
|
||||||
|
: size;
|
||||||
|
}
|
||||||
|
|
||||||
bool OverlayWidget::videoShown() const {
|
bool OverlayWidget::videoShown() const {
|
||||||
return _streamed && !_streamed->instance.info().video.cover.isNull();
|
return _streamed && !_streamed->instance.info().video.cover.isNull();
|
||||||
}
|
}
|
||||||
|
@ -439,7 +472,7 @@ bool OverlayWidget::videoShown() const {
|
||||||
QSize OverlayWidget::videoSize() const {
|
QSize OverlayWidget::videoSize() const {
|
||||||
Expects(videoShown());
|
Expects(videoShown());
|
||||||
|
|
||||||
return _streamed->instance.info().video.size;
|
return flipSizeByRotation(_streamed->instance.info().video.size);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OverlayWidget::videoIsGifv() const {
|
bool OverlayWidget::videoIsGifv() const {
|
||||||
|
@ -498,7 +531,7 @@ QImage OverlayWidget::videoFrameForDirectPaint() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OverlayWidget::documentContentShown() const {
|
bool OverlayWidget::documentContentShown() const {
|
||||||
return _doc && (!_current.isNull() || videoShown());
|
return _doc && (!_staticContent.isNull() || videoShown());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OverlayWidget::documentBubbleShown() const {
|
bool OverlayWidget::documentBubbleShown() const {
|
||||||
|
@ -506,7 +539,7 @@ bool OverlayWidget::documentBubbleShown() const {
|
||||||
|| (_doc
|
|| (_doc
|
||||||
&& !_themePreviewShown
|
&& !_themePreviewShown
|
||||||
&& !_streamed
|
&& !_streamed
|
||||||
&& _current.isNull());
|
&& _staticContent.isNull());
|
||||||
}
|
}
|
||||||
|
|
||||||
void OverlayWidget::clearStreaming(bool savePosition) {
|
void OverlayWidget::clearStreaming(bool savePosition) {
|
||||||
|
@ -630,8 +663,10 @@ void OverlayWidget::updateControls() {
|
||||||
|| (_doc
|
|| (_doc
|
||||||
&& _doc->filepath(DocumentData::FilePathResolve::Checked).isEmpty()
|
&& _doc->filepath(DocumentData::FilePathResolve::Checked).isEmpty()
|
||||||
&& !_doc->loading());
|
&& !_doc->loading());
|
||||||
_saveNav = myrtlrect(width() - st::mediaviewIconSize.width() * 2, height() - st::mediaviewIconSize.height(), st::mediaviewIconSize.width(), st::mediaviewIconSize.height());
|
_saveNav = myrtlrect(width() - st::mediaviewIconSize.width() * 3, height() - st::mediaviewIconSize.height(), st::mediaviewIconSize.width(), st::mediaviewIconSize.height());
|
||||||
_saveNavIcon = style::centerrect(_saveNav, st::mediaviewSave);
|
_saveNavIcon = style::centerrect(_saveNav, st::mediaviewSave);
|
||||||
|
_rotateNav = myrtlrect(width() - st::mediaviewIconSize.width() * 2, height() - st::mediaviewIconSize.height(), st::mediaviewIconSize.width(), st::mediaviewIconSize.height());
|
||||||
|
_rotateNavIcon = style::centerrect(_rotateNav, st::mediaviewRotate);
|
||||||
_moreNav = myrtlrect(width() - st::mediaviewIconSize.width(), height() - st::mediaviewIconSize.height(), st::mediaviewIconSize.width(), st::mediaviewIconSize.height());
|
_moreNav = myrtlrect(width() - st::mediaviewIconSize.width(), height() - st::mediaviewIconSize.height(), st::mediaviewIconSize.width(), st::mediaviewIconSize.height());
|
||||||
_moreNavIcon = style::centerrect(_moreNav, st::mediaviewMore);
|
_moreNavIcon = style::centerrect(_moreNav, st::mediaviewMore);
|
||||||
|
|
||||||
|
@ -826,6 +861,7 @@ bool OverlayWidget::updateControlsAnimation(crl::time now) {
|
||||||
+ (_over == OverRightNav ? _rightNav : _rightNavIcon)
|
+ (_over == OverRightNav ? _rightNav : _rightNavIcon)
|
||||||
+ (_over == OverClose ? _closeNav : _closeNavIcon)
|
+ (_over == OverClose ? _closeNav : _closeNavIcon)
|
||||||
+ _saveNavIcon
|
+ _saveNavIcon
|
||||||
|
+ _rotateNavIcon
|
||||||
+ _moreNavIcon
|
+ _moreNavIcon
|
||||||
+ _headerNav
|
+ _headerNav
|
||||||
+ _nameNav
|
+ _nameNav
|
||||||
|
@ -848,6 +884,15 @@ void OverlayWidget::updateCursor() {
|
||||||
: (_over == OverNone ? style::cur_default : style::cur_pointer));
|
: (_over == OverNone ? style::cur_default : style::cur_pointer));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int OverlayWidget::contentRotation() const {
|
||||||
|
if (!_streamed) {
|
||||||
|
return _rotation;
|
||||||
|
}
|
||||||
|
return (_rotation + (_streamed
|
||||||
|
? _streamed->instance.info().video.rotation
|
||||||
|
: 0)) % 360;
|
||||||
|
}
|
||||||
|
|
||||||
QRect OverlayWidget::contentRect() const {
|
QRect OverlayWidget::contentRect() const {
|
||||||
return { _x, _y, _w, _h };
|
return { _x, _y, _w, _h };
|
||||||
}
|
}
|
||||||
|
@ -1410,12 +1455,9 @@ void OverlayWidget::onOverview() {
|
||||||
void OverlayWidget::onCopy() {
|
void OverlayWidget::onCopy() {
|
||||||
_dropdown->hideAnimated(Ui::DropdownMenu::HideOption::IgnoreShow);
|
_dropdown->hideAnimated(Ui::DropdownMenu::HideOption::IgnoreShow);
|
||||||
if (_doc) {
|
if (_doc) {
|
||||||
if (videoShown()) {
|
QGuiApplication::clipboard()->setImage(videoShown()
|
||||||
QGuiApplication::clipboard()->setImage(
|
? transformVideoFrame(videoFrame())
|
||||||
transformVideoFrame(videoFrame()));
|
: transformStaticContent(_staticContent));
|
||||||
} else if (!_current.isNull()) {
|
|
||||||
QGuiApplication::clipboard()->setPixmap(_current);
|
|
||||||
}
|
|
||||||
} else if (_photo && _photo->loaded()) {
|
} else if (_photo && _photo->loaded()) {
|
||||||
QGuiApplication::clipboard()->setPixmap(_photo->large()->pix(fileOrigin()));
|
QGuiApplication::clipboard()->setPixmap(_photo->large()->pix(fileOrigin()));
|
||||||
}
|
}
|
||||||
|
@ -1899,6 +1941,7 @@ void OverlayWidget::displayPhoto(not_null<PhotoData*> photo, HistoryItem *item)
|
||||||
_doc = nullptr;
|
_doc = nullptr;
|
||||||
_fullScreenVideo = false;
|
_fullScreenVideo = false;
|
||||||
_photo = photo;
|
_photo = photo;
|
||||||
|
_rotation = _photo->owner().mediaRotation().get(_photo);
|
||||||
_radial.stop();
|
_radial.stop();
|
||||||
|
|
||||||
refreshMediaViewer();
|
refreshMediaViewer();
|
||||||
|
@ -1907,10 +1950,13 @@ void OverlayWidget::displayPhoto(not_null<PhotoData*> photo, HistoryItem *item)
|
||||||
_zoom = 0;
|
_zoom = 0;
|
||||||
_zoomToScreen = _zoomToDefault = 0;
|
_zoomToScreen = _zoomToDefault = 0;
|
||||||
_blurred = true;
|
_blurred = true;
|
||||||
_current = QPixmap();
|
_staticContent = QPixmap();
|
||||||
_down = OverNone;
|
_down = OverNone;
|
||||||
_w = style::ConvertScale(photo->width());
|
const auto size = style::ConvertScale(flipSizeByRotation(QSize(
|
||||||
_h = style::ConvertScale(photo->height());
|
photo->width(),
|
||||||
|
photo->height())));
|
||||||
|
_w = size.width();
|
||||||
|
_h = size.height();
|
||||||
contentSizeChanged();
|
contentSizeChanged();
|
||||||
refreshFromLabel(item);
|
refreshFromLabel(item);
|
||||||
_photo->download(fileOrigin());
|
_photo->download(fileOrigin());
|
||||||
|
@ -1948,10 +1994,11 @@ void OverlayWidget::displayDocument(
|
||||||
moveToScreen();
|
moveToScreen();
|
||||||
}
|
}
|
||||||
_fullScreenVideo = false;
|
_fullScreenVideo = false;
|
||||||
_current = QPixmap();
|
_staticContent = QPixmap();
|
||||||
clearStreaming(_doc != doc);
|
clearStreaming(_doc != doc);
|
||||||
destroyThemePreview();
|
destroyThemePreview();
|
||||||
_doc = doc;
|
_doc = doc;
|
||||||
|
_rotation = _doc ? _doc->owner().mediaRotation().get(_doc) : 0;
|
||||||
_themeCloudData = cloud;
|
_themeCloudData = cloud;
|
||||||
_photo = nullptr;
|
_photo = nullptr;
|
||||||
_radial.stop();
|
_radial.stop();
|
||||||
|
@ -1960,9 +2007,9 @@ void OverlayWidget::displayDocument(
|
||||||
if (_doc) {
|
if (_doc) {
|
||||||
if (_doc->sticker()) {
|
if (_doc->sticker()) {
|
||||||
if (const auto image = _doc->getStickerLarge()) {
|
if (const auto image = _doc->getStickerLarge()) {
|
||||||
_current = image->pix(fileOrigin());
|
_staticContent = image->pix(fileOrigin());
|
||||||
} else if (_doc->hasThumbnail()) {
|
} else if (_doc->hasThumbnail()) {
|
||||||
_current = _doc->thumbnail()->pixBlurred(
|
_staticContent = _doc->thumbnail()->pixBlurred(
|
||||||
fileOrigin(),
|
fileOrigin(),
|
||||||
_doc->dimensions.width(),
|
_doc->dimensions.width(),
|
||||||
_doc->dimensions.height());
|
_doc->dimensions.height());
|
||||||
|
@ -1981,7 +2028,7 @@ void OverlayWidget::displayDocument(
|
||||||
if (location.accessEnable()) {
|
if (location.accessEnable()) {
|
||||||
const auto &path = location.name();
|
const auto &path = location.name();
|
||||||
if (QImageReader(path).canRead()) {
|
if (QImageReader(path).canRead()) {
|
||||||
_current = PrepareStaticImage(path);
|
_staticContent = PrepareStaticImage(path);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
location.accessDisable();
|
location.accessDisable();
|
||||||
|
@ -2045,10 +2092,12 @@ void OverlayWidget::displayDocument(
|
||||||
_docIconRect = myrtlrect(_docRect.x() + st::mediaviewFilePadding, _docRect.y() + st::mediaviewFilePadding, st::mediaviewFileIconSize, st::mediaviewFileIconSize);
|
_docIconRect = myrtlrect(_docRect.x() + st::mediaviewFilePadding, _docRect.y() + st::mediaviewFilePadding, st::mediaviewFileIconSize, st::mediaviewFileIconSize);
|
||||||
} else if (_themePreviewShown) {
|
} else if (_themePreviewShown) {
|
||||||
updateThemePreviewGeometry();
|
updateThemePreviewGeometry();
|
||||||
} else if (!_current.isNull()) {
|
} else if (!_staticContent.isNull()) {
|
||||||
_current.setDevicePixelRatio(cRetinaFactor());
|
_staticContent.setDevicePixelRatio(cRetinaFactor());
|
||||||
_w = style::ConvertScale(_current.width());
|
const auto size = style::ConvertScale(
|
||||||
_h = style::ConvertScale(_current.height());
|
flipSizeByRotation(_staticContent.size()));
|
||||||
|
_w = size.width();
|
||||||
|
_h = size.height();
|
||||||
} else if (videoShown()) {
|
} else if (videoShown()) {
|
||||||
const auto contentSize = style::ConvertScale(videoSize());
|
const auto contentSize = style::ConvertScale(videoSize());
|
||||||
_w = contentSize.width();
|
_w = contentSize.width();
|
||||||
|
@ -2175,7 +2224,7 @@ void OverlayWidget::initStreamingThumbnail() {
|
||||||
const auto h = size.height();
|
const auto h = size.height();
|
||||||
const auto options = VideoThumbOptions(_doc);
|
const auto options = VideoThumbOptions(_doc);
|
||||||
const auto goodOptions = (options & ~Images::Option::Blurred);
|
const auto goodOptions = (options & ~Images::Option::Blurred);
|
||||||
_current = (useGood
|
_staticContent = (useGood
|
||||||
? good
|
? good
|
||||||
: useThumb
|
: useThumb
|
||||||
? thumb
|
? thumb
|
||||||
|
@ -2188,20 +2237,24 @@ void OverlayWidget::initStreamingThumbnail() {
|
||||||
useGood ? goodOptions : options,
|
useGood ? goodOptions : options,
|
||||||
w / cIntRetinaFactor(),
|
w / cIntRetinaFactor(),
|
||||||
h / cIntRetinaFactor());
|
h / cIntRetinaFactor());
|
||||||
_current.setDevicePixelRatio(cRetinaFactor());
|
_staticContent.setDevicePixelRatio(cRetinaFactor());
|
||||||
}
|
}
|
||||||
|
|
||||||
void OverlayWidget::streamingReady(Streaming::Information &&info) {
|
void OverlayWidget::streamingReady(Streaming::Information &&info) {
|
||||||
if (videoShown()) {
|
if (videoShown()) {
|
||||||
const auto contentSize = style::ConvertScale(videoSize());
|
applyVideoSize();
|
||||||
if (contentSize != QSize(_width, _height)) {
|
}
|
||||||
update(contentRect());
|
update(contentRect());
|
||||||
_w = contentSize.width();
|
}
|
||||||
_h = contentSize.height();
|
|
||||||
contentSizeChanged();
|
void OverlayWidget::applyVideoSize() {
|
||||||
}
|
const auto contentSize = style::ConvertScale(videoSize());
|
||||||
|
if (contentSize != QSize(_width, _height)) {
|
||||||
|
update(contentRect());
|
||||||
|
_w = contentSize.width();
|
||||||
|
_h = contentSize.height();
|
||||||
|
contentSizeChanged();
|
||||||
}
|
}
|
||||||
this->update(contentRect());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OverlayWidget::createStreamingObjects() {
|
bool OverlayWidget::createStreamingObjects() {
|
||||||
|
@ -2234,20 +2287,32 @@ bool OverlayWidget::createStreamingObjects() {
|
||||||
QImage OverlayWidget::transformVideoFrame(QImage frame) const {
|
QImage OverlayWidget::transformVideoFrame(QImage frame) const {
|
||||||
Expects(videoShown());
|
Expects(videoShown());
|
||||||
|
|
||||||
if (_streamed->instance.info().video.rotation != 0) {
|
const auto rotation = contentRotation();
|
||||||
|
if (rotation != 0) {
|
||||||
auto transform = QTransform();
|
auto transform = QTransform();
|
||||||
transform.rotate(_streamed->instance.info().video.rotation);
|
transform.rotate(rotation);
|
||||||
frame = frame.transformed(transform);
|
frame = frame.transformed(transform);
|
||||||
}
|
}
|
||||||
if (frame.size() != _streamed->instance.info().video.size) {
|
const auto requiredSize = videoSize();
|
||||||
|
if (frame.size() != requiredSize) {
|
||||||
frame = frame.scaled(
|
frame = frame.scaled(
|
||||||
_streamed->instance.info().video.size,
|
requiredSize,
|
||||||
Qt::IgnoreAspectRatio,
|
Qt::IgnoreAspectRatio,
|
||||||
Qt::SmoothTransformation);
|
Qt::SmoothTransformation);
|
||||||
}
|
}
|
||||||
return frame;
|
return frame;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QImage OverlayWidget::transformStaticContent(QPixmap content) const {
|
||||||
|
auto image = content.toImage();
|
||||||
|
if (!_rotation) {
|
||||||
|
return image;
|
||||||
|
}
|
||||||
|
auto transform = QTransform();
|
||||||
|
transform.rotate(_rotation);
|
||||||
|
return image.transformed(transform);
|
||||||
|
}
|
||||||
|
|
||||||
void OverlayWidget::handleStreamingUpdate(Streaming::Update &&update) {
|
void OverlayWidget::handleStreamingUpdate(Streaming::Update &&update) {
|
||||||
using namespace Streaming;
|
using namespace Streaming;
|
||||||
|
|
||||||
|
@ -2420,6 +2485,25 @@ void OverlayWidget::playbackControlsToPictureInPicture() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void OverlayWidget::playbackControlsRotate() {
|
||||||
|
if (_photo) {
|
||||||
|
auto &storage = _photo->owner().mediaRotation();
|
||||||
|
storage.set(_photo, storage.get(_photo) - 90);
|
||||||
|
_rotation = storage.get(_photo);
|
||||||
|
redisplayContent();
|
||||||
|
} else if (_doc) {
|
||||||
|
auto &storage = _doc->owner().mediaRotation();
|
||||||
|
storage.set(_doc, storage.get(_doc) - 90);
|
||||||
|
_rotation = storage.get(_doc);
|
||||||
|
if (videoShown()) {
|
||||||
|
applyVideoSize();
|
||||||
|
update(contentRect());
|
||||||
|
} else {
|
||||||
|
redisplayContent();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void OverlayWidget::playbackPauseResume() {
|
void OverlayWidget::playbackPauseResume() {
|
||||||
Expects(_streamed != nullptr);
|
Expects(_streamed != nullptr);
|
||||||
|
|
||||||
|
@ -2449,7 +2533,9 @@ void OverlayWidget::restartAtSeekPosition(crl::time position) {
|
||||||
|
|
||||||
if (videoShown()) {
|
if (videoShown()) {
|
||||||
_streamed->instance.saveFrameToCover();
|
_streamed->instance.saveFrameToCover();
|
||||||
_current = Images::PixmapFast(transformVideoFrame(videoFrame()));
|
const auto saved = base::take(_rotation);
|
||||||
|
_staticContent = Images::PixmapFast(transformVideoFrame(videoFrame()));
|
||||||
|
_rotation = saved;
|
||||||
update(contentRect());
|
update(contentRect());
|
||||||
}
|
}
|
||||||
auto options = Streaming::PlaybackOptions();
|
auto options = Streaming::PlaybackOptions();
|
||||||
|
@ -2625,18 +2711,18 @@ void OverlayWidget::validatePhotoImage(Image *image, bool blurred) {
|
||||||
image->load(fileOrigin());
|
image->load(fileOrigin());
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
} else if (!_current.isNull() && (blurred || !_blurred)) {
|
} else if (!_staticContent.isNull() && (blurred || !_blurred)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const auto w = _width * cIntRetinaFactor();
|
const auto w = _width * cIntRetinaFactor();
|
||||||
const auto h = _height * cIntRetinaFactor();
|
const auto h = _height * cIntRetinaFactor();
|
||||||
_current = image->pixNoCache(
|
_staticContent = image->pixNoCache(
|
||||||
fileOrigin(),
|
fileOrigin(),
|
||||||
w,
|
w,
|
||||||
h,
|
h,
|
||||||
Images::Option::Smooth
|
Images::Option::Smooth
|
||||||
| (blurred ? Images::Option::Blurred : Images::Option(0)));
|
| (blurred ? Images::Option::Blurred : Images::Option(0)));
|
||||||
_current.setDevicePixelRatio(cRetinaFactor());
|
_staticContent.setDevicePixelRatio(cRetinaFactor());
|
||||||
_blurred = blurred;
|
_blurred = blurred;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2645,7 +2731,7 @@ void OverlayWidget::validatePhotoCurrentImage() {
|
||||||
validatePhotoImage(_photo->thumbnail(), true);
|
validatePhotoImage(_photo->thumbnail(), true);
|
||||||
validatePhotoImage(_photo->thumbnailSmall(), true);
|
validatePhotoImage(_photo->thumbnailSmall(), true);
|
||||||
validatePhotoImage(_photo->thumbnailInline(), true);
|
validatePhotoImage(_photo->thumbnailInline(), true);
|
||||||
if (_current.isNull()
|
if (_staticContent.isNull()
|
||||||
&& _peer
|
&& _peer
|
||||||
&& !_msgid
|
&& !_msgid
|
||||||
&& _peer->userpicLoaded()
|
&& _peer->userpicLoaded()
|
||||||
|
@ -2654,7 +2740,7 @@ void OverlayWidget::validatePhotoCurrentImage() {
|
||||||
Images::Create(_peer->userpicLocation()).get(),
|
Images::Create(_peer->userpicLocation()).get(),
|
||||||
true);
|
true);
|
||||||
}
|
}
|
||||||
if (_current.isNull()) {
|
if (_staticContent.isNull()) {
|
||||||
_photo->loadThumbnailSmall(fileOrigin());
|
_photo->loadThumbnailSmall(fileOrigin());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2697,14 +2783,7 @@ void OverlayWidget::paintEvent(QPaintEvent *e) {
|
||||||
if (videoShown()) {
|
if (videoShown()) {
|
||||||
paintTransformedVideoFrame(p);
|
paintTransformedVideoFrame(p);
|
||||||
} else {
|
} else {
|
||||||
if ((!_doc || !_doc->getStickerLarge())
|
paintTransformedStaticContent(p);
|
||||||
&& (_current.isNull() || _current.hasAlpha())) {
|
|
||||||
p.fillRect(rect, _transparentBrush);
|
|
||||||
}
|
|
||||||
if (!_current.isNull()) {
|
|
||||||
PainterHighQualityEnabler hq(p);
|
|
||||||
p.drawPixmap(rect, _current);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto radial = _radial.animating();
|
const auto radial = _radial.animating();
|
||||||
|
@ -2834,6 +2913,13 @@ void OverlayWidget::paintEvent(QPaintEvent *e) {
|
||||||
st::mediaviewSave.paintInCenter(p, _saveNavIcon);
|
st::mediaviewSave.paintInCenter(p, _saveNavIcon);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// rotate button
|
||||||
|
if (_rotateNavIcon.intersects(r)) {
|
||||||
|
auto o = overLevel(OverRotate);
|
||||||
|
p.setOpacity((o * st::mediaviewIconOverOpacity + (1 - o) * st::mediaviewIconOpacity) * co);
|
||||||
|
st::mediaviewRotate.paintInCenter(p, _rotateNavIcon);
|
||||||
|
}
|
||||||
|
|
||||||
// more area
|
// more area
|
||||||
if (_moreNavIcon.intersects(r)) {
|
if (_moreNavIcon.intersects(r)) {
|
||||||
auto o = overLevel(OverMore);
|
auto o = overLevel(OverMore);
|
||||||
|
@ -2927,46 +3013,52 @@ void OverlayWidget::paintTransformedVideoFrame(Painter &p) {
|
||||||
|
|
||||||
const auto rect = contentRect();
|
const auto rect = contentRect();
|
||||||
const auto image = videoFrameForDirectPaint();
|
const auto image = videoFrameForDirectPaint();
|
||||||
//if (_fullScreenVideo) {
|
|
||||||
// const auto fill = rect.intersected(this->rect());
|
|
||||||
// PaintImageProfile(p, image, rect, fill);
|
|
||||||
//} else {
|
|
||||||
const auto rotation = _streamed->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);
|
PainterHighQualityEnabler hq(p);
|
||||||
if (rotation) {
|
|
||||||
p.save();
|
const auto rotation = contentRotation();
|
||||||
p.rotate(rotation);
|
if (UsePainterRotation(rotation)) {
|
||||||
}
|
if (rotation) {
|
||||||
p.drawImage(rotated(rect, rotation), image);
|
p.save();
|
||||||
if (rotation) {
|
p.rotate(rotation);
|
||||||
p.restore();
|
}
|
||||||
|
p.drawImage(RotatedRect(rect, rotation), image);
|
||||||
|
if (rotation) {
|
||||||
|
p.restore();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
p.drawImage(rect, transformVideoFrame(image));
|
||||||
}
|
}
|
||||||
if (_streamed->instance.player().ready()) {
|
if (_streamed->instance.player().ready()) {
|
||||||
_streamed->instance.markFrameShown();
|
_streamed->instance.markFrameShown();
|
||||||
}
|
}
|
||||||
//}
|
}
|
||||||
|
|
||||||
|
void OverlayWidget::paintTransformedStaticContent(Painter &p) {
|
||||||
|
const auto rect = contentRect();
|
||||||
|
|
||||||
|
PainterHighQualityEnabler hq(p);
|
||||||
|
if ((!_doc || !_doc->getStickerLarge())
|
||||||
|
&& (_staticContent.isNull()
|
||||||
|
|| _staticContent.hasAlpha())) {
|
||||||
|
p.fillRect(rect, _transparentBrush);
|
||||||
|
}
|
||||||
|
if (_staticContent.isNull()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto rotation = contentRotation();
|
||||||
|
if (UsePainterRotation(rotation)) {
|
||||||
|
if (rotation) {
|
||||||
|
p.save();
|
||||||
|
p.rotate(rotation);
|
||||||
|
}
|
||||||
|
p.drawPixmap(RotatedRect(rect, rotation), _staticContent);
|
||||||
|
if (rotation) {
|
||||||
|
p.restore();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
p.drawImage(rect, transformStaticContent(_staticContent));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void OverlayWidget::paintRadialLoading(
|
void OverlayWidget::paintRadialLoading(
|
||||||
|
@ -3438,6 +3530,7 @@ void OverlayWidget::mousePressEvent(QMouseEvent *e) {
|
||||||
|| _over == OverDate
|
|| _over == OverDate
|
||||||
|| _over == OverHeader
|
|| _over == OverHeader
|
||||||
|| _over == OverSave
|
|| _over == OverSave
|
||||||
|
|| _over == OverRotate
|
||||||
|| _over == OverIcon
|
|| _over == OverIcon
|
||||||
|| _over == OverMore
|
|| _over == OverMore
|
||||||
|| _over == OverClose
|
|| _over == OverClose
|
||||||
|
@ -3515,6 +3608,7 @@ void OverlayWidget::updateOverRect(OverState state) {
|
||||||
case OverName: update(_nameNav); break;
|
case OverName: update(_nameNav); break;
|
||||||
case OverDate: update(_dateNav); break;
|
case OverDate: update(_dateNav); break;
|
||||||
case OverSave: update(_saveNavIcon); break;
|
case OverSave: update(_saveNavIcon); break;
|
||||||
|
case OverRotate: update(_rotateNavIcon); break;
|
||||||
case OverIcon: update(_docIconRect); break;
|
case OverIcon: update(_docIconRect); break;
|
||||||
case OverHeader: update(_headerNav); break;
|
case OverHeader: update(_headerNav); break;
|
||||||
case OverClose: update(_closeNav); break;
|
case OverClose: update(_closeNav); break;
|
||||||
|
@ -3608,6 +3702,8 @@ void OverlayWidget::updateOver(QPoint pos) {
|
||||||
updateOverState(OverHeader);
|
updateOverState(OverHeader);
|
||||||
} else if (_saveVisible && _saveNav.contains(pos)) {
|
} else if (_saveVisible && _saveNav.contains(pos)) {
|
||||||
updateOverState(OverSave);
|
updateOverState(OverSave);
|
||||||
|
} else if (_rotateNav.contains(pos)) {
|
||||||
|
updateOverState(OverRotate);
|
||||||
} else if (_doc && documentBubbleShown() && _docIconRect.contains(pos)) {
|
} else if (_doc && documentBubbleShown() && _docIconRect.contains(pos)) {
|
||||||
updateOverState(OverIcon);
|
updateOverState(OverIcon);
|
||||||
} else if (_moreNav.contains(pos)) {
|
} else if (_moreNav.contains(pos)) {
|
||||||
|
@ -3650,6 +3746,8 @@ void OverlayWidget::mouseReleaseEvent(QMouseEvent *e) {
|
||||||
onOverview();
|
onOverview();
|
||||||
} else if (_over == OverSave && _down == OverSave) {
|
} else if (_over == OverSave && _down == OverSave) {
|
||||||
onDownload();
|
onDownload();
|
||||||
|
} else if (_over == OverRotate && _down == OverRotate) {
|
||||||
|
playbackControlsRotate();
|
||||||
} else if (_over == OverIcon && _down == OverIcon) {
|
} else if (_over == OverIcon && _down == OverIcon) {
|
||||||
onDocClick();
|
onDocClick();
|
||||||
} else if (_over == OverMore && _down == OverMore) {
|
} else if (_over == OverMore && _down == OverMore) {
|
||||||
|
@ -3872,7 +3970,7 @@ void OverlayWidget::setVisibleHook(bool visible) {
|
||||||
clearStreaming();
|
clearStreaming();
|
||||||
destroyThemePreview();
|
destroyThemePreview();
|
||||||
_radial.stop();
|
_radial.stop();
|
||||||
_current = QPixmap();
|
_staticContent = QPixmap();
|
||||||
_themePreview = nullptr;
|
_themePreview = nullptr;
|
||||||
_themeApply.destroyDelayed();
|
_themeApply.destroyDelayed();
|
||||||
_themeCancel.destroyDelayed();
|
_themeCancel.destroyDelayed();
|
||||||
|
|
|
@ -140,6 +140,7 @@ private:
|
||||||
OverName,
|
OverName,
|
||||||
OverDate,
|
OverDate,
|
||||||
OverSave,
|
OverSave,
|
||||||
|
OverRotate,
|
||||||
OverMore,
|
OverMore,
|
||||||
OverIcon,
|
OverIcon,
|
||||||
OverVideo,
|
OverVideo,
|
||||||
|
@ -180,6 +181,7 @@ private:
|
||||||
void playbackControlsToFullScreen() override;
|
void playbackControlsToFullScreen() override;
|
||||||
void playbackControlsFromFullScreen() override;
|
void playbackControlsFromFullScreen() override;
|
||||||
void playbackControlsToPictureInPicture() override;
|
void playbackControlsToPictureInPicture() override;
|
||||||
|
void playbackControlsRotate() override;
|
||||||
void playbackPauseResume();
|
void playbackPauseResume();
|
||||||
void playbackToggleFullScreen();
|
void playbackToggleFullScreen();
|
||||||
void playbackPauseOnCall();
|
void playbackPauseOnCall();
|
||||||
|
@ -283,7 +285,8 @@ private:
|
||||||
void documentUpdated(DocumentData *doc);
|
void documentUpdated(DocumentData *doc);
|
||||||
void changingMsgId(not_null<HistoryItem*> row, MsgId newId);
|
void changingMsgId(not_null<HistoryItem*> row, MsgId newId);
|
||||||
|
|
||||||
QRect contentRect() const;
|
[[nodiscard]] int contentRotation() const;
|
||||||
|
[[nodiscard]] QRect contentRect() const;
|
||||||
void contentSizeChanged();
|
void contentSizeChanged();
|
||||||
|
|
||||||
// Radial animation interface.
|
// Radial animation interface.
|
||||||
|
@ -325,21 +328,27 @@ private:
|
||||||
void validatePhotoImage(Image *image, bool blurred);
|
void validatePhotoImage(Image *image, bool blurred);
|
||||||
void validatePhotoCurrentImage();
|
void validatePhotoCurrentImage();
|
||||||
|
|
||||||
|
[[nodiscard]] QSize flipSizeByRotation(QSize size) const;
|
||||||
|
|
||||||
|
void applyVideoSize();
|
||||||
[[nodiscard]] bool videoShown() const;
|
[[nodiscard]] bool videoShown() const;
|
||||||
[[nodiscard]] QSize videoSize() const;
|
[[nodiscard]] QSize videoSize() const;
|
||||||
[[nodiscard]] bool videoIsGifv() const;
|
[[nodiscard]] bool videoIsGifv() const;
|
||||||
[[nodiscard]] QImage videoFrame() const;
|
[[nodiscard]] QImage videoFrame() const;
|
||||||
[[nodiscard]] QImage videoFrameForDirectPaint() const;
|
[[nodiscard]] QImage videoFrameForDirectPaint() const;
|
||||||
[[nodiscard]] QImage transformVideoFrame(QImage frame) const;
|
[[nodiscard]] QImage transformVideoFrame(QImage frame) const;
|
||||||
|
[[nodiscard]] QImage transformStaticContent(QPixmap content) const;
|
||||||
[[nodiscard]] bool documentContentShown() const;
|
[[nodiscard]] bool documentContentShown() const;
|
||||||
[[nodiscard]] bool documentBubbleShown() const;
|
[[nodiscard]] bool documentBubbleShown() const;
|
||||||
void paintTransformedVideoFrame(Painter &p);
|
void paintTransformedVideoFrame(Painter &p);
|
||||||
|
void paintTransformedStaticContent(Painter &p);
|
||||||
void clearStreaming(bool savePosition = true);
|
void clearStreaming(bool savePosition = true);
|
||||||
|
|
||||||
QBrush _transparentBrush;
|
QBrush _transparentBrush;
|
||||||
|
|
||||||
PhotoData *_photo = nullptr;
|
PhotoData *_photo = nullptr;
|
||||||
DocumentData *_doc = nullptr;
|
DocumentData *_doc = nullptr;
|
||||||
|
int _rotation = 0;
|
||||||
std::unique_ptr<SharedMedia> _sharedMedia;
|
std::unique_ptr<SharedMedia> _sharedMedia;
|
||||||
std::optional<SharedMediaWithLastSlice> _sharedMediaData;
|
std::optional<SharedMediaWithLastSlice> _sharedMediaData;
|
||||||
std::optional<SharedMediaWithLastSlice::Key> _sharedMediaDataKey;
|
std::optional<SharedMediaWithLastSlice::Key> _sharedMediaDataKey;
|
||||||
|
@ -351,7 +360,7 @@ private:
|
||||||
QRect _closeNav, _closeNavIcon;
|
QRect _closeNav, _closeNavIcon;
|
||||||
QRect _leftNav, _leftNavIcon, _rightNav, _rightNavIcon;
|
QRect _leftNav, _leftNavIcon, _rightNav, _rightNavIcon;
|
||||||
QRect _headerNav, _nameNav, _dateNav;
|
QRect _headerNav, _nameNav, _dateNav;
|
||||||
QRect _saveNav, _saveNavIcon, _moreNav, _moreNavIcon;
|
QRect _rotateNav, _rotateNavIcon, _saveNav, _saveNavIcon, _moreNav, _moreNavIcon;
|
||||||
bool _leftNavVisible = false;
|
bool _leftNavVisible = false;
|
||||||
bool _rightNavVisible = false;
|
bool _rightNavVisible = false;
|
||||||
bool _saveVisible = false;
|
bool _saveVisible = false;
|
||||||
|
@ -382,7 +391,7 @@ private:
|
||||||
QPoint _mStart;
|
QPoint _mStart;
|
||||||
bool _pressed = false;
|
bool _pressed = false;
|
||||||
int32 _dragging = 0;
|
int32 _dragging = 0;
|
||||||
QPixmap _current;
|
QPixmap _staticContent;
|
||||||
bool _blurred = true;
|
bool _blurred = true;
|
||||||
|
|
||||||
std::unique_ptr<Streamed> _streamed;
|
std::unique_ptr<Streamed> _streamed;
|
||||||
|
|
|
@ -223,6 +223,10 @@ void PlaybackControls::showMenu() {
|
||||||
addSpeed(1.75);
|
addSpeed(1.75);
|
||||||
addSpeed(2.);
|
addSpeed(2.);
|
||||||
_menu.emplace(this, st::mediaviewControlsPopupMenu);
|
_menu.emplace(this, st::mediaviewControlsPopupMenu);
|
||||||
|
_menu->addAction("Rotate video", [=] {
|
||||||
|
_delegate->playbackControlsRotate();
|
||||||
|
});
|
||||||
|
_menu->addSeparator();
|
||||||
_menu->addAction(
|
_menu->addAction(
|
||||||
tr::lng_mediaview_playback_speed(tr::now),
|
tr::lng_mediaview_playback_speed(tr::now),
|
||||||
std::move(submenu));
|
std::move(submenu));
|
||||||
|
|
|
@ -46,6 +46,7 @@ public:
|
||||||
virtual void playbackControlsToFullScreen() = 0;
|
virtual void playbackControlsToFullScreen() = 0;
|
||||||
virtual void playbackControlsFromFullScreen() = 0;
|
virtual void playbackControlsFromFullScreen() = 0;
|
||||||
virtual void playbackControlsToPictureInPicture() = 0;
|
virtual void playbackControlsToPictureInPicture() = 0;
|
||||||
|
virtual void playbackControlsRotate() = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
PlaybackControls(QWidget *parent, not_null<Delegate*> delegate);
|
PlaybackControls(QWidget *parent, not_null<Delegate*> delegate);
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit 1888853b52291c32ce5bbca7212e67c262c76cee
|
Subproject commit 628d3b9ab6443acd352617ac78cc3131ba41dbbc
|