Fix crash in the EditCaptionBox.

This commit is contained in:
John Preston 2020-06-04 16:35:26 +04:00
parent fbb2bae99f
commit e1d36cfd50
2 changed files with 54 additions and 66 deletions

View File

@ -72,31 +72,21 @@ EditCaptionBox::EditCaptionBox(
_isAllowedEditMedia = item->media()->allowsEditMedia(); _isAllowedEditMedia = item->media()->allowsEditMedia();
_isAlbum = !item->groupId().empty(); _isAlbum = !item->groupId().empty();
QSize dimensions; auto dimensions = QSize();
auto image = (Image*)nullptr;
const auto media = item->media(); const auto media = item->media();
if (const auto photo = media->photo()) { if (const auto photo = media->photo()) {
_photoMedia = photo->createMediaView(); _photoMedia = photo->createMediaView();
_photoMedia->wanted(PhotoSize::Large, _msgId); _photoMedia->wanted(PhotoSize::Large, _msgId);
image = _photoMedia->image(PhotoSize::Large);
if (!image) {
image = _photoMedia->image(PhotoSize::Thumbnail);
if (!image) {
image = _photoMedia->image(PhotoSize::Small);
if (!image) {
image = _photoMedia->thumbnailInline();
}
}
}
dimensions = _photoMedia->size(PhotoSize::Large); dimensions = _photoMedia->size(PhotoSize::Large);
if (dimensions.isEmpty()) {
dimensions = QSize(1, 1);
}
_photo = true; _photo = true;
} else if (const auto document = media->document()) { } else if (const auto document = media->document()) {
_documentMedia = document->createMediaView(); _documentMedia = document->createMediaView();
_documentMedia->thumbnailWanted(_msgId); _documentMedia->thumbnailWanted(_msgId);
image = _documentMedia->thumbnail(); dimensions = _documentMedia->thumbnail()
dimensions = image ? _documentMedia->thumbnail()->size()
? image->size()
: document->dimensions; : document->dimensions;
if (document->isAnimation()) { if (document->isAnimation()) {
_gifw = style::ConvertScale(document->dimensions.width()); _gifw = style::ConvertScale(document->dimensions.width());
@ -107,24 +97,42 @@ EditCaptionBox::EditCaptionBox(
} else { } else {
_doc = true; _doc = true;
} }
} else {
Unexpected("Photo or document should be set.");
} }
const auto editData = PrepareEditText(item); const auto editData = PrepareEditText(item);
if (!_animated const auto computeImage = [=] {
&& (dimensions.isEmpty() if (_documentMedia) {
|| _documentMedia return _documentMedia->thumbnail();
|| (!_photoMedia && !image))) { } else if (const auto large = _photoMedia->image(PhotoSize::Large)) {
if (!image) { return large;
_thumbw = 0; } else if (const auto thumbnail = _photoMedia->image(
PhotoSize::Thumbnail)) {
return thumbnail;
} else if (const auto small = _photoMedia->image(PhotoSize::Small)) {
return small;
} else { } else {
const auto tw = image->width(), th = image->height(); return _photoMedia->thumbnailInline();
}
};
if (!_animated && _documentMedia) {
if (dimensions.isEmpty()) {
_thumbw = 0;
_thumbnailImageLoaded = true;
} else {
const auto tw = dimensions.width(), th = dimensions.height();
if (tw > th) { if (tw > th) {
_thumbw = (tw * st::msgFileThumbSize) / th; _thumbw = (tw * st::msgFileThumbSize) / th;
} else { } else {
_thumbw = st::msgFileThumbSize; _thumbw = st::msgFileThumbSize;
} }
_thumbnailImage = image;
_refreshThumbnail = [=] { _refreshThumbnail = [=] {
const auto image = computeImage();
if (!image) {
return;
}
const auto options = Images::Option::Smooth const auto options = Images::Option::Smooth
| Images::Option::RoundedSmall | Images::Option::RoundedSmall
| Images::Option::RoundedTopLeft | Images::Option::RoundedTopLeft
@ -138,7 +146,9 @@ EditCaptionBox::EditCaptionBox(
options, options,
st::msgFileThumbSize, st::msgFileThumbSize,
st::msgFileThumbSize)); st::msgFileThumbSize));
_thumbnailImageLoaded = true;
}; };
_refreshThumbnail();
} }
if (_documentMedia) { if (_documentMedia) {
@ -151,13 +161,7 @@ EditCaptionBox::EditCaptionBox(
_isAudio = document->isVoiceMessage() _isAudio = document->isVoiceMessage()
|| document->isAudioFile(); || document->isAudioFile();
} }
if (_refreshThumbnail) {
_refreshThumbnail();
}
} else { } else {
if (!image && !_photoMedia) {
image = Image::BlankMedia();
}
auto maxW = 0, maxH = 0; auto maxW = 0, maxH = 0;
const auto limitW = st::sendMediaPreviewSize; const auto limitW = st::sendMediaPreviewSize;
auto limitH = std::min(st::confirmMaxHeight, _gifh ? _gifh : INT_MAX); auto limitH = std::min(st::confirmMaxHeight, _gifh ? _gifh : INT_MAX);
@ -175,35 +179,38 @@ EditCaptionBox::EditCaptionBox(
maxH = limitH; maxH = limitH;
} }
} }
_thumbnailImage = image;
_refreshThumbnail = [=] { _refreshThumbnail = [=] {
const auto image = computeImage();
const auto use = image ? image : Image::BlankMedia().get();
const auto options = Images::Option::Smooth const auto options = Images::Option::Smooth
| Images::Option::Blurred; | Images::Option::Blurred;
_thumb = image->pixNoCache( _thumb = use->pixNoCache(
maxW * cIntRetinaFactor(), maxW * cIntRetinaFactor(),
maxH * cIntRetinaFactor(), maxH * cIntRetinaFactor(),
options, options,
maxW, maxW,
maxH); maxH);
_thumbnailImageLoaded = true;
}; };
prepareStreamedPreview(); prepareStreamedPreview();
} else { } else {
Assert(_photoMedia != nullptr);
maxW = dimensions.width(); maxW = dimensions.width();
maxH = dimensions.height(); maxH = dimensions.height();
_thumbnailImage = image;
_refreshThumbnail = [=] { _refreshThumbnail = [=] {
const auto photo = _photoMedia const auto image = computeImage();
? _photoMedia->image(Data::PhotoSize::Large) const auto photo = _photoMedia->image(Data::PhotoSize::Large);
: nullptr;
const auto use = photo const auto use = photo
? photo ? photo
: _thumbnailImage : image
? _thumbnailImage ? image
: Image::BlankMedia().get(); : Image::BlankMedia().get();
const auto options = Images::Option::Smooth const auto options = Images::Option::Smooth
| ((_photoMedia && !photo) | (photo
? Images::Option::Blurred ? Images::Option(0)
: Images::Option(0)); : Images::Option::Blurred);
_thumbnailImageLoaded = (photo != nullptr);
_thumb = use->pixNoCache( _thumb = use->pixNoCache(
maxW * cIntRetinaFactor(), maxW * cIntRetinaFactor(),
maxH * cIntRetinaFactor(), maxH * cIntRetinaFactor(),
@ -276,35 +283,17 @@ EditCaptionBox::EditCaptionBox(
scaleThumbDown(); scaleThumbDown();
} }
Assert(_animated || _photo || _doc); Assert(_animated || _photo || _doc);
Assert(_thumbnailImageLoaded || _refreshThumbnail);
_thumbnailImageLoaded = _photoMedia
? (_photoMedia->image(Data::PhotoSize::Large) != nullptr)
: _thumbnailImage
? true
: _documentMedia
? !_documentMedia->owner()->hasThumbnail()
: true;
if (!_thumbnailImageLoaded) { if (!_thumbnailImageLoaded) {
subscribe(_controller->session().downloaderTaskFinished(), [=] { subscribe(_controller->session().downloaderTaskFinished(), [=] {
if (_thumbnailImageLoaded) { if (_thumbnailImageLoaded
|| (_photoMedia && !_photoMedia->image(PhotoSize::Large))
|| (_documentMedia && !_documentMedia->thumbnail())) {
return; return;
} else if (!_thumbnailImage
&& _photoMedia
&& _photoMedia->image(PhotoSize::Large)) {
_thumbnailImage = _photoMedia->image(PhotoSize::Large);
} else if (!_thumbnailImage
&& _documentMedia
&& _documentMedia->owner()->hasThumbnail()) {
_thumbnailImage = _documentMedia->thumbnail();
}
if (_thumbnailImage) {
_thumbnailImageLoaded = !_photoMedia
|| _photoMedia->image(PhotoSize::Large);
if (_thumbnailImageLoaded) {
_refreshThumbnail();
update();
}
} }
_refreshThumbnail();
update();
}); });
} }
_field.create( _field.create(

View File

@ -108,7 +108,6 @@ private:
FullMsgId _msgId; FullMsgId _msgId;
std::shared_ptr<Data::PhotoMedia> _photoMedia; std::shared_ptr<Data::PhotoMedia> _photoMedia;
std::shared_ptr<Data::DocumentMedia> _documentMedia; std::shared_ptr<Data::DocumentMedia> _documentMedia;
Image *_thumbnailImage = nullptr;
bool _thumbnailImageLoaded = false; bool _thumbnailImageLoaded = false;
Fn<void()> _refreshThumbnail; Fn<void()> _refreshThumbnail;
bool _animated = false; bool _animated = false;