Add button to download video files.

Fixes #5781.
This commit is contained in:
John Preston 2019-03-15 14:15:04 +04:00
parent f2a7cf5c64
commit bc2b0f8392
4 changed files with 89 additions and 23 deletions

View File

@ -10,6 +10,7 @@ using "dialogs/dialogs.style";
using "ui/widgets/widgets.style"; using "ui/widgets/widgets.style";
minPhotoSize: 100px; minPhotoSize: 100px;
minVideoSize: 160px;
maxMediaSize: 430px; maxMediaSize: 430px;
maxStickerSize: 256px; maxStickerSize: 256px;
maxGifSize: 320px; maxGifSize: 320px;
@ -552,3 +553,9 @@ historyGroupAboutBulletSkip: 16px;
historyGroupAboutHeaderSkip: 10px; historyGroupAboutHeaderSkip: 10px;
historyGroupAboutTextSkip: 10px; historyGroupAboutTextSkip: 10px;
historyGroupAboutSkip: 8px; historyGroupAboutSkip: 8px;
historyVideoDownloadSize: 36px;
historyVideoCancel: icon {{ "playlist_cancel", historyFileThumbIconFg }};
historyVideoCancelSelected: icon {{ "playlist_cancel", historyFileThumbIconFgSelected }};
historyVideoDownload: icon {{ "playlist_download", historyFileThumbIconFg }};
historyVideoDownloadSelected: icon {{ "playlist_download", historyFileThumbIconFgSelected }};

View File

@ -40,6 +40,7 @@ HistoryVideo::HistoryVideo(
setDocumentLinks(_data, realParent); setDocumentLinks(_data, realParent);
setStatusSize(FileStatusSizeReady); setStatusSize(FileStatusSizeReady);
_downloadSize = formatSizeText(_data->size);
_data->loadThumbnail(realParent->fullId()); _data->loadThumbnail(realParent->fullId());
} }
@ -100,10 +101,10 @@ QSize HistoryVideo::countOptimalSize() {
_thumbw = qMax(tw, 1); _thumbw = qMax(tw, 1);
_thumbh = qMax(th, 1); _thumbh = qMax(th, 1);
auto minWidth = qMax(st::minPhotoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); auto minWidth = qMax(st::minVideoSize, _parent->infoWidth() + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x()));
minWidth = qMax(minWidth, documentMaxStatusWidth(_data) + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x())); minWidth = qMax(minWidth, documentMaxStatusWidth(_data) + 2 * (st::msgDateImgDelta + st::msgDateImgPadding.x()));
auto maxWidth = qMax(_thumbw, minWidth); auto maxWidth = qMax(_thumbw, minWidth);
auto minHeight = qMax(th, st::minPhotoSize); auto minHeight = qMax(th, st::minVideoSize);
if (_parent->hasBubble() && !_caption.isEmpty()) { if (_parent->hasBubble() && !_caption.isEmpty()) {
const auto captionw = maxWidth const auto captionw = maxWidth
- st::msgPadding.left() - st::msgPadding.left()
@ -143,6 +144,12 @@ QSize HistoryVideo::countCurrentSize(int newWidth) {
return { newWidth, newHeight }; return { newWidth, newHeight };
} }
bool HistoryVideo::downloadInCorner() const {
return _data->canBeStreamed()
&& !_data->inappPlaybackFailed()
&& IsServerMsgId(_parent->data()->id);
}
void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms) const { void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms) const {
if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return;
@ -152,6 +159,7 @@ void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, crl
auto paintx = 0, painty = 0, paintw = width(), painth = height(); auto paintx = 0, painty = 0, paintw = width(), painth = height();
bool bubble = _parent->hasBubble(); bool bubble = _parent->hasBubble();
const auto cornerDownload = downloadInCorner();
int captionw = paintw - st::msgPadding.left() - st::msgPadding.right(); int captionw = paintw - st::msgPadding.left() - st::msgPadding.right();
@ -227,12 +235,8 @@ void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, crl
p.drawEllipse(inner); p.drawEllipse(inner);
} }
if (!selected && _animation) { const auto icon = [&]() -> const style::icon * {
p.setOpacity(1); if (!cornerDownload && (_data->loading() || _data->uploading())) {
}
auto icon = [&]() -> const style::icon * {
if (_data->loading() || _data->uploading()) {
return &(selected ? st::historyFileThumbCancelSelected : st::historyFileThumbCancel); return &(selected ? st::historyFileThumbCancelSelected : st::historyFileThumbCancel);
} else if (!IsServerMsgId(_parent->data()->id)) { } else if (!IsServerMsgId(_parent->data()->id)) {
return nullptr; return nullptr;
@ -244,18 +248,12 @@ void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, crl
if (icon) { if (icon) {
icon->paintInCenter(p, inner); icon->paintInCenter(p, inner);
} }
if (radial) { if (radial && !cornerDownload) {
QRect rinner(inner.marginsRemoved(QMargins(st::msgFileRadialLine, st::msgFileRadialLine, st::msgFileRadialLine, st::msgFileRadialLine))); QRect rinner(inner.marginsRemoved(QMargins(st::msgFileRadialLine, st::msgFileRadialLine, st::msgFileRadialLine, st::msgFileRadialLine)));
_animation->radial.draw(p, rinner, st::msgFileRadialLine, selected ? st::historyFileThumbRadialFgSelected : st::historyFileThumbRadialFg); _animation->radial.draw(p, rinner, st::msgFileRadialLine, selected ? st::historyFileThumbRadialFgSelected : st::historyFileThumbRadialFg);
} }
auto statusX = paintx + st::msgDateImgDelta + st::msgDateImgPadding.x(), statusY = painty + st::msgDateImgDelta + st::msgDateImgPadding.y(); drawCornerStatus(p, selected);
auto statusW = st::normalFont->width(_statusText) + 2 * st::msgDateImgPadding.x();
auto statusH = st::normalFont->height + 2 * st::msgDateImgPadding.y();
App::roundRect(p, rtlrect(statusX - st::msgDateImgPadding.x(), statusY - st::msgDateImgPadding.y(), statusW, statusH, width()), selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners);
p.setFont(st::normalFont);
p.setPen(st::msgDateImgFg);
p.drawTextLeft(statusX, statusY, width(), _statusText, statusW - 2 * st::msgDateImgPadding.x());
// date // date
if (!_caption.isEmpty()) { if (!_caption.isEmpty()) {
@ -273,6 +271,56 @@ void HistoryVideo::draw(Painter &p, const QRect &r, TextSelection selection, crl
} }
} }
void HistoryVideo::drawCornerStatus(Painter &p, bool selected) const {
const auto padding = st::msgDateImgPadding;
const auto radial = _animation && _animation->radial.animating();
const auto cornerDownload = downloadInCorner() && !_data->loaded();
const auto addWidth = cornerDownload ? (st::historyVideoDownloadSize + 2 * padding.y() - padding.x()) : 0;
const auto downloadWidth = cornerDownload ? st::normalFont->width(_downloadSize) : 0;
const auto statusX = st::msgDateImgDelta + padding.x(), statusY = st::msgDateImgDelta + padding.y();
const auto statusW = std::max(downloadWidth, st::normalFont->width(_statusText)) + 2 * padding.x() + addWidth;
const auto statusH = cornerDownload ? (st::historyVideoDownloadSize + 2 * padding.y()) : (st::normalFont->height + 2 * padding.y());
const auto around = rtlrect(statusX - padding.x(), statusY - padding.y(), statusW, statusH, width());
App::roundRect(p, around, selected ? st::msgDateImgBgSelected : st::msgDateImgBg, selected ? DateSelectedCorners : DateCorners);
p.setFont(st::normalFont);
p.setPen(st::msgDateImgFg);
p.drawTextLeft(statusX + addWidth, statusY, width(), _statusText, statusW - 2 * padding.x());
if (cornerDownload) {
p.drawTextLeft(statusX + addWidth, statusY + statusH - 2 * padding.y() - st::normalFont->height, width(), _downloadSize, statusW - 2 * padding.x());
const auto inner = QRect(statusX + padding.y() - padding.x(), statusY, st::historyVideoDownloadSize, st::historyVideoDownloadSize);
const auto icon = [&]() -> const style::icon * {
if (_data->loading()) {
return &(selected ? st::historyVideoCancelSelected : st::historyVideoCancel);
}
return &(selected ? st::historyVideoDownloadSelected : st::historyVideoDownload);
}();
if (icon) {
icon->paintInCenter(p, inner);
}
if (radial) {
QRect rinner(inner.marginsRemoved(QMargins(st::msgFileRadialLine, st::msgFileRadialLine, st::msgFileRadialLine, st::msgFileRadialLine)));
_animation->radial.draw(p, rinner, st::msgFileRadialLine, selected ? st::historyFileThumbRadialFgSelected : st::historyFileThumbRadialFg);
}
}
}
TextState HistoryVideo::cornerStatusTextState(
QPoint point,
StateRequest request) const {
auto result = TextState(_parent);
if (!downloadInCorner() || _data->loaded()) {
return result;
}
const auto padding = st::msgDateImgPadding;
const auto addWidth = st::historyVideoDownloadSize + 2 * padding.y() - padding.x();
const auto statusX = st::msgDateImgDelta + padding.x(), statusY = st::msgDateImgDelta + padding.y();
const auto inner = QRect(statusX + padding.y() - padding.x(), statusY, st::historyVideoDownloadSize, st::historyVideoDownloadSize);
if (inner.contains(point)) {
result.link = _data->loading() ? _cancell : _savel;
}
return result;
}
TextState HistoryVideo::textState(QPoint point, StateRequest request) const { TextState HistoryVideo::textState(QPoint point, StateRequest request) const {
if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) { if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) {
return {}; return {};
@ -299,8 +347,11 @@ TextState HistoryVideo::textState(QPoint point, StateRequest request) const {
} }
painth -= st::mediaCaptionSkip; painth -= st::mediaCaptionSkip;
} }
if (const auto state = cornerStatusTextState(point, request); state.link) {
return state;
}
if (QRect(paintx, painty, paintw, painth).contains(point)) { if (QRect(paintx, painty, paintw, painth).contains(point)) {
if (_data->loading() || _data->uploading()) { if (!downloadInCorner() && (_data->loading() || _data->uploading())) {
result.link = _cancell; result.link = _cancell;
} else if (!IsServerMsgId(_parent->data()->id)) { } else if (!IsServerMsgId(_parent->data()->id)) {
} else if (_data->loaded() || _data->canBePlayed()) { } else if (_data->loaded() || _data->canBePlayed()) {
@ -556,9 +607,9 @@ void HistoryVideo::updateStatusText() const {
statusSize = FileStatusSizeFailed; statusSize = FileStatusSizeFailed;
} else if (_data->uploading()) { } else if (_data->uploading()) {
statusSize = _data->uploadingData->offset; statusSize = _data->uploadingData->offset;
} else if (_data->loading()) { } else if (!downloadInCorner() && _data->loading()) {
statusSize = _data->loadOffset(); statusSize = _data->loadOffset();
} else if (_data->loaded()) { } else if (_data->canBePlayed()) {
statusSize = FileStatusSizeLoaded; statusSize = FileStatusSizeLoaded;
} else { } else {
statusSize = FileStatusSizeReady; statusSize = FileStatusSizeReady;

View File

@ -73,9 +73,15 @@ protected:
bool dataLoaded() const override; bool dataLoaded() const override;
private: private:
QSize countOptimalSize() override; [[nodiscard]] QSize countOptimalSize() override;
QSize countCurrentSize(int newWidth) override; [[nodiscard]] QSize countCurrentSize(int newWidth) override;
QSize countOptimalDimensions() const; [[nodiscard]] QSize countOptimalDimensions() const;
[[nodiscard]] bool downloadInCorner() const;
void drawCornerStatus(Painter &p, bool selected) const;
[[nodiscard]] TextState cornerStatusTextState(
QPoint point,
StateRequest request) const;
void validateGroupedCache( void validateGroupedCache(
const QRect &geometry, const QRect &geometry,
@ -91,4 +97,6 @@ private:
int _thumbh = 1; int _thumbh = 1;
Text _caption; Text _caption;
QString _downloadSize;
}; };

View File

@ -822,7 +822,7 @@ void OverlayWidget::step_radial(crl::time ms, bool timer) {
const auto streamVideo = ready && _doc->canBePlayed(); const auto streamVideo = ready && _doc->canBePlayed();
const auto tryOpenImage = ready && (_doc->size < App::kImageSizeLimit); const auto tryOpenImage = ready && (_doc->size < App::kImageSizeLimit);
if (ready && ((tryOpenImage && !_radial.animating()) || streamVideo)) { if (ready && ((tryOpenImage && !_radial.animating()) || streamVideo)) {
_streamingStartPaused = true; _streamingStartPaused = false;
if (streamVideo) { if (streamVideo) {
redisplayContent(); redisplayContent();
} else { } else {