mirror of https://github.com/procxx/kepka.git
inline bots gif / photo preview / sending done
This commit is contained in:
parent
14839e7afe
commit
5830e0f657
|
@ -113,6 +113,9 @@ namespace {
|
||||||
typedef QHash<PhotoData*, LastPhotosList::iterator> LastPhotosMap;
|
typedef QHash<PhotoData*, LastPhotosList::iterator> LastPhotosMap;
|
||||||
LastPhotosMap lastPhotosMap;
|
LastPhotosMap lastPhotosMap;
|
||||||
|
|
||||||
|
typedef QMap<FileLoader*, InlineResult*> InlineResultLoaders;
|
||||||
|
InlineResultLoaders inlineResultLoaders;
|
||||||
|
|
||||||
style::color _msgServiceBg;
|
style::color _msgServiceBg;
|
||||||
style::color _msgServiceSelectBg;
|
style::color _msgServiceSelectBg;
|
||||||
style::color _historyScrollBarColor;
|
style::color _historyScrollBarColor;
|
||||||
|
@ -1459,12 +1462,12 @@ namespace App {
|
||||||
delete convert->uploadingData;
|
delete convert->uploadingData;
|
||||||
convert->uploadingData = 0;
|
convert->uploadingData = 0;
|
||||||
}
|
}
|
||||||
convert->access = access;
|
if (date) {
|
||||||
if (!convert->date && date) {
|
convert->access = access;
|
||||||
convert->date = date;
|
convert->date = date;
|
||||||
convert->thumb = thumb;
|
if (convert->thumb->isNull() && !thumb->isNull()) convert->thumb = thumb;
|
||||||
convert->medium = medium;
|
if (convert->medium->isNull() && !medium->isNull()) convert->medium = medium;
|
||||||
convert->full = full;
|
if (convert->full->isNull() && !full->isNull()) convert->full = full;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
PhotosData::const_iterator i = ::photosData.constFind(photo);
|
PhotosData::const_iterator i = ::photosData.constFind(photo);
|
||||||
|
@ -1479,12 +1482,12 @@ namespace App {
|
||||||
::photosData.insert(photo, result);
|
::photosData.insert(photo, result);
|
||||||
} else {
|
} else {
|
||||||
result = i.value();
|
result = i.value();
|
||||||
if (result != convert && !result->date && date) {
|
if (result != convert && date) {
|
||||||
result->access = access;
|
result->access = access;
|
||||||
result->date = date;
|
result->date = date;
|
||||||
result->thumb = thumb;
|
if (result->thumb->isNull() && !thumb->isNull()) result->thumb = thumb;
|
||||||
result->medium = medium;
|
if (result->medium->isNull() && !medium->isNull()) result->medium = medium;
|
||||||
result->full = full;
|
if (result->full->isNull() && !full->isNull()) result->full = full;
|
||||||
}
|
}
|
||||||
inLastIter = lastPhotosMap.find(result);
|
inLastIter = lastPhotosMap.find(result);
|
||||||
}
|
}
|
||||||
|
@ -1520,13 +1523,15 @@ namespace App {
|
||||||
convert->id = video;
|
convert->id = video;
|
||||||
convert->status = FileReady;
|
convert->status = FileReady;
|
||||||
}
|
}
|
||||||
convert->access = access;
|
if (date) {
|
||||||
if (!convert->date && date) {
|
convert->access = access;
|
||||||
convert->date = date;
|
convert->date = date;
|
||||||
|
if (convert->thumb->isNull() && !thumb->isNull()) {
|
||||||
|
convert->thumb = thumb;
|
||||||
|
}
|
||||||
convert->duration = duration;
|
convert->duration = duration;
|
||||||
convert->w = w;
|
convert->w = w;
|
||||||
convert->h = h;
|
convert->h = h;
|
||||||
convert->thumb = thumb;
|
|
||||||
convert->dc = dc;
|
convert->dc = dc;
|
||||||
convert->size = size;
|
convert->size = size;
|
||||||
}
|
}
|
||||||
|
@ -1542,13 +1547,15 @@ namespace App {
|
||||||
::videosData.insert(video, result);
|
::videosData.insert(video, result);
|
||||||
} else {
|
} else {
|
||||||
result = i.value();
|
result = i.value();
|
||||||
if (result != convert && !result->date && date) {
|
if (result != convert && date) {
|
||||||
result->access = access;
|
result->access = access;
|
||||||
result->date = date;
|
result->date = date;
|
||||||
result->duration = duration;
|
result->duration = duration;
|
||||||
result->w = w;
|
result->w = w;
|
||||||
result->h = h;
|
result->h = h;
|
||||||
result->thumb = thumb;
|
if (result->thumb->isNull() && !thumb->isNull()) {
|
||||||
|
result->thumb = thumb;
|
||||||
|
}
|
||||||
result->dc = dc;
|
result->dc = dc;
|
||||||
result->size = size;
|
result->size = size;
|
||||||
}
|
}
|
||||||
|
@ -1574,8 +1581,8 @@ namespace App {
|
||||||
convert->id = audio;
|
convert->id = audio;
|
||||||
convert->status = FileReady;
|
convert->status = FileReady;
|
||||||
}
|
}
|
||||||
convert->access = access;
|
if (date) {
|
||||||
if (!convert->date && date) {
|
convert->access = access;
|
||||||
convert->date = date;
|
convert->date = date;
|
||||||
convert->mime = mime;
|
convert->mime = mime;
|
||||||
convert->duration = duration;
|
convert->duration = duration;
|
||||||
|
@ -1594,7 +1601,7 @@ namespace App {
|
||||||
::audiosData.insert(audio, result);
|
::audiosData.insert(audio, result);
|
||||||
} else {
|
} else {
|
||||||
result = i.value();
|
result = i.value();
|
||||||
if (result != convert && !result->date && date) {
|
if (result != convert && date) {
|
||||||
result->access = access;
|
result->access = access;
|
||||||
result->date = date;
|
result->date = date;
|
||||||
result->mime = mime;
|
result->mime = mime;
|
||||||
|
@ -1622,6 +1629,7 @@ namespace App {
|
||||||
if (i != ::documentsData.cend() && i.value() == convert) {
|
if (i != ::documentsData.cend() && i.value() == convert) {
|
||||||
::documentsData.erase(i);
|
::documentsData.erase(i);
|
||||||
}
|
}
|
||||||
|
Local::copyStickerImage(mediaKey(DocumentFileLocation, convert->dc, convert->id), mediaKey(DocumentFileLocation, dc, document));
|
||||||
convert->id = document;
|
convert->id = document;
|
||||||
convert->status = FileReady;
|
convert->status = FileReady;
|
||||||
if (cSavedGifs().indexOf(convert) >= 0) { // id changed
|
if (cSavedGifs().indexOf(convert) >= 0) { // id changed
|
||||||
|
@ -1629,35 +1637,21 @@ namespace App {
|
||||||
}
|
}
|
||||||
sentSticker = !!convert->sticker();
|
sentSticker = !!convert->sticker();
|
||||||
}
|
}
|
||||||
convert->access = access;
|
if (date) {
|
||||||
if (!convert->date && date) {
|
convert->access = access;
|
||||||
convert->date = date;
|
convert->date = date;
|
||||||
convert->setattributes(attributes);
|
convert->setattributes(attributes);
|
||||||
convert->mime = mime;
|
convert->mime = mime;
|
||||||
convert->thumb = thumb;
|
|
||||||
convert->dc = dc;
|
|
||||||
convert->size = size;
|
|
||||||
convert->recountIsImage();
|
|
||||||
} else {
|
|
||||||
if (!thumb->isNull() && (convert->thumb->isNull() || convert->thumb->width() < thumb->width() || convert->thumb->height() < thumb->height())) {
|
if (!thumb->isNull() && (convert->thumb->isNull() || convert->thumb->width() < thumb->width() || convert->thumb->height() < thumb->height())) {
|
||||||
convert->thumb = thumb;
|
convert->thumb = thumb;
|
||||||
}
|
}
|
||||||
if (convert->sticker() && !attributes.isEmpty() && (convert->sticker()->alt.isEmpty() || convert->sticker()->set.type() == mtpc_inputStickerSetEmpty)) {
|
convert->dc = dc;
|
||||||
for (QVector<MTPDocumentAttribute>::const_iterator i = attributes.cbegin(), e = attributes.cend(); i != e; ++i) {
|
convert->size = size;
|
||||||
if (i->type() == mtpc_documentAttributeSticker) {
|
convert->recountIsImage();
|
||||||
const MTPDdocumentAttributeSticker &d(i->c_documentAttributeSticker());
|
if (convert->sticker() && convert->sticker()->loc.isNull() && !thumbLocation.isNull()) {
|
||||||
if (d.valt.c_string().v.length() > 0) {
|
convert->sticker()->loc = thumbLocation;
|
||||||
convert->sticker()->alt = qs(d.valt);
|
|
||||||
convert->sticker()->set = d.vstickerset;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (convert->sticker() && convert->sticker()->loc.isNull() && !thumbLocation.isNull()) {
|
|
||||||
convert->sticker()->loc = thumbLocation;
|
|
||||||
}
|
|
||||||
|
|
||||||
const FileLocation &loc(convert->location(true));
|
const FileLocation &loc(convert->location(true));
|
||||||
if (!loc.isEmpty()) {
|
if (!loc.isEmpty()) {
|
||||||
Local::writeFileLocation(mediaKey(DocumentFileLocation, convert->dc, convert->id), loc);
|
Local::writeFileLocation(mediaKey(DocumentFileLocation, convert->dc, convert->id), loc);
|
||||||
|
@ -1676,34 +1670,19 @@ namespace App {
|
||||||
::documentsData.insert(document, result);
|
::documentsData.insert(document, result);
|
||||||
} else {
|
} else {
|
||||||
result = i.value();
|
result = i.value();
|
||||||
if (result != convert) {
|
if (result != convert && date) {
|
||||||
if (!result->date && date) {
|
result->access = access;
|
||||||
result->access = access;
|
result->date = date;
|
||||||
result->date = date;
|
result->setattributes(attributes);
|
||||||
result->setattributes(attributes);
|
result->mime = mime;
|
||||||
result->mime = mime;
|
if (!thumb->isNull() && (result->thumb->isNull() || result->thumb->width() < thumb->width() || result->thumb->height() < thumb->height())) {
|
||||||
result->thumb = thumb;
|
result->thumb = thumb;
|
||||||
result->dc = dc;
|
}
|
||||||
result->size = size;
|
result->dc = dc;
|
||||||
result->recountIsImage();
|
result->size = size;
|
||||||
} else {
|
result->recountIsImage();
|
||||||
if (!thumb->isNull() && (result->thumb->isNull() || result->thumb->width() < thumb->width() || result->thumb->height() < thumb->height())) {
|
if (result->sticker() && result->sticker()->loc.isNull() && !thumbLocation.isNull()) {
|
||||||
result->thumb = thumb;
|
result->sticker()->loc = thumbLocation;
|
||||||
}
|
|
||||||
if (result->sticker() && !attributes.isEmpty() && (result->sticker()->alt.isEmpty() || result->sticker()->set.type() == mtpc_inputStickerSetEmpty)) {
|
|
||||||
for (QVector<MTPDocumentAttribute>::const_iterator i = attributes.cbegin(), e = attributes.cend(); i != e; ++i) {
|
|
||||||
if (i->type() == mtpc_documentAttributeSticker) {
|
|
||||||
const MTPDdocumentAttributeSticker &d(i->c_documentAttributeSticker());
|
|
||||||
if (d.valt.c_string().v.length() > 0) {
|
|
||||||
result->sticker()->alt = qs(d.valt);
|
|
||||||
result->sticker()->set = d.vstickerset;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (result->sticker() && result->sticker()->loc.isNull() && !thumbLocation.isNull()) {
|
|
||||||
result->sticker()->loc = thumbLocation;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2522,6 +2501,19 @@ namespace App {
|
||||||
if (changeInMin) App::main()->updateMutedIn(changeInMin);
|
if (changeInMin) App::main()->updateMutedIn(changeInMin);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void regInlineResultLoader(FileLoader *loader, InlineResult *result) {
|
||||||
|
::inlineResultLoaders.insert(loader, result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void unregInlineResultLoader(FileLoader *loader) {
|
||||||
|
::inlineResultLoaders.remove(loader);
|
||||||
|
}
|
||||||
|
|
||||||
|
InlineResult *inlineResultFromLoader(FileLoader *loader) {
|
||||||
|
InlineResultLoaders::const_iterator i = ::inlineResultLoaders.find(loader);
|
||||||
|
return (i == ::inlineResultLoaders.cend()) ? 0 : i.value();
|
||||||
|
}
|
||||||
|
|
||||||
inline void insertReplyMarkup(ChannelId channelId, MsgId msgId, const ReplyMarkup &markup) {
|
inline void insertReplyMarkup(ChannelId channelId, MsgId msgId, const ReplyMarkup &markup) {
|
||||||
if (channelId == NoChannel) {
|
if (channelId == NoChannel) {
|
||||||
replyMarkups.insert(msgId, markup);
|
replyMarkups.insert(msgId, markup);
|
||||||
|
|
|
@ -246,6 +246,10 @@ namespace App {
|
||||||
void unregMuted(PeerData *peer);
|
void unregMuted(PeerData *peer);
|
||||||
void updateMuted();
|
void updateMuted();
|
||||||
|
|
||||||
|
void regInlineResultLoader(FileLoader *loader, InlineResult *result);
|
||||||
|
void unregInlineResultLoader(FileLoader *loader);
|
||||||
|
InlineResult *inlineResultFromLoader(FileLoader *loader);
|
||||||
|
|
||||||
void feedReplyMarkup(ChannelId channelId, MsgId msgId, const MTPReplyMarkup &markup);
|
void feedReplyMarkup(ChannelId channelId, MsgId msgId, const MTPReplyMarkup &markup);
|
||||||
void clearReplyMarkup(ChannelId channelId, MsgId msgId);
|
void clearReplyMarkup(ChannelId channelId, MsgId msgId);
|
||||||
const ReplyMarkup &replyMarkup(ChannelId channelId, MsgId msgId);
|
const ReplyMarkup &replyMarkup(ChannelId channelId, MsgId msgId);
|
||||||
|
|
|
@ -1432,15 +1432,61 @@ void StickerPanInner::mouseReleaseEvent(QMouseEvent *e) {
|
||||||
if (row < _inlineRows.size() && col < _inlineRows.at(row).items.size()) {
|
if (row < _inlineRows.size() && col < _inlineRows.at(row).items.size()) {
|
||||||
if (down) {
|
if (down) {
|
||||||
if (down->type() == qstr("SendInlineItemLink") && e->button() == Qt::LeftButton) {
|
if (down->type() == qstr("SendInlineItemLink") && e->button() == Qt::LeftButton) {
|
||||||
DocumentData *doc = _inlineRows.at(row).items.at(col)->document();
|
LayoutInlineItem *item = _inlineRows.at(row).items.at(col);
|
||||||
if (!doc) return;
|
PhotoData *photo = item->photo();
|
||||||
|
DocumentData *doc = item->document();
|
||||||
if (doc->loaded()) {
|
InlineResult *result = item->result();
|
||||||
emit selected(doc);
|
if (doc) {
|
||||||
} else if (doc->loading()) {
|
if (doc->loaded()) {
|
||||||
doc->cancel();
|
emit selected(doc);
|
||||||
} else {
|
} else if (doc->loading()) {
|
||||||
DocumentOpenLink::doOpen(doc, ActionOnLoadNone);
|
doc->cancel();
|
||||||
|
} else {
|
||||||
|
DocumentOpenLink::doOpen(doc, ActionOnLoadNone);
|
||||||
|
}
|
||||||
|
} else if (photo) {
|
||||||
|
if (photo->medium->loaded() || photo->thumb->loaded()) {
|
||||||
|
emit selected(photo);
|
||||||
|
} else if (!photo->medium->loading()) {
|
||||||
|
photo->thumb->loadEvenCancelled();
|
||||||
|
photo->medium->loadEvenCancelled();
|
||||||
|
}
|
||||||
|
} else if (result) {
|
||||||
|
if (result->type == qstr("gif")) {
|
||||||
|
if (result->doc) {
|
||||||
|
if (result->doc->loaded()) {
|
||||||
|
emit selected(result, _inlineBot);
|
||||||
|
} else if (result->doc->loading()) {
|
||||||
|
result->doc->cancel();
|
||||||
|
} else {
|
||||||
|
DocumentOpenLink::doOpen(result->doc, ActionOnLoadNone);
|
||||||
|
}
|
||||||
|
} else if (result->loaded()) {
|
||||||
|
emit selected(result, _inlineBot);
|
||||||
|
} else if (result->loading()) {
|
||||||
|
result->cancelFile();
|
||||||
|
Ui::repaintInlineItem(item);
|
||||||
|
} else {
|
||||||
|
result->saveFile(QString(), LoadFromCloudOrLocal, false);
|
||||||
|
Ui::repaintInlineItem(item);
|
||||||
|
}
|
||||||
|
} else if (result->type == qstr("photo")) {
|
||||||
|
if (result->photo) {
|
||||||
|
if (result->photo->medium->loaded() || result->photo->thumb->loaded()) {
|
||||||
|
emit selected(result, _inlineBot);
|
||||||
|
} else if (!result->photo->medium->loading()) {
|
||||||
|
result->photo->thumb->loadEvenCancelled();
|
||||||
|
result->photo->medium->loadEvenCancelled();
|
||||||
|
}
|
||||||
|
} else if (result->thumb->loaded()) {
|
||||||
|
emit selected(result, _inlineBot);
|
||||||
|
} else if (!result->thumb->loading()) {
|
||||||
|
result->thumb->loadEvenCancelled();
|
||||||
|
Ui::repaintInlineItem(item);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
emit selected(result, _inlineBot);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
down->onClick(e->button());
|
down->onClick(e->button());
|
||||||
|
@ -1558,7 +1604,7 @@ void StickerPanInner::clearSelection(bool fast) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void StickerPanInner::hideFinish() {
|
void StickerPanInner::hideFinish() {
|
||||||
clearInlineRows();
|
clearInlineRows(false);
|
||||||
for (GifLayouts::const_iterator i = _gifLayouts.cbegin(), e = _gifLayouts.cend(); i != e; ++i) {
|
for (GifLayouts::const_iterator i = _gifLayouts.cbegin(), e = _gifLayouts.cend(); i != e; ++i) {
|
||||||
i.value()->document()->forget();
|
i.value()->document()->forget();
|
||||||
}
|
}
|
||||||
|
@ -1607,13 +1653,15 @@ void StickerPanInner::inlineRowsAddItem(DocumentData *savedGif, InlineResult *re
|
||||||
}
|
}
|
||||||
if (!layout) return;
|
if (!layout) return;
|
||||||
|
|
||||||
inlineRowFinalize(row, sumWidth, layout->fullLine());
|
if (inlineRowFinalize(row, sumWidth, layout->fullLine())) {
|
||||||
|
layout->setPosition(_inlineRows.size() * MatrixRowShift);
|
||||||
|
}
|
||||||
row.items.push_back(layout);
|
row.items.push_back(layout);
|
||||||
sumWidth += layout->maxWidth();
|
sumWidth += layout->maxWidth();
|
||||||
}
|
}
|
||||||
|
|
||||||
void StickerPanInner::inlineRowFinalize(InlineRow &row, int32 &sumWidth, bool force) {
|
bool StickerPanInner::inlineRowFinalize(InlineRow &row, int32 &sumWidth, bool force) {
|
||||||
if (row.items.isEmpty()) return;
|
if (row.items.isEmpty()) return false;
|
||||||
|
|
||||||
bool full = (row.items.size() >= SavedGifsMaxPerRow);
|
bool full = (row.items.size() >= SavedGifsMaxPerRow);
|
||||||
bool big = (sumWidth >= st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft - (row.items.size() - 1) * st::inlineResultsSkip);
|
bool big = (sumWidth >= st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft - (row.items.size() - 1) * st::inlineResultsSkip);
|
||||||
|
@ -1622,12 +1670,14 @@ void StickerPanInner::inlineRowFinalize(InlineRow &row, int32 &sumWidth, bool fo
|
||||||
row = InlineRow();
|
row = InlineRow();
|
||||||
row.items.reserve(SavedGifsMaxPerRow);
|
row.items.reserve(SavedGifsMaxPerRow);
|
||||||
sumWidth = 0;
|
sumWidth = 0;
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void StickerPanInner::refreshSavedGifs() {
|
void StickerPanInner::refreshSavedGifs() {
|
||||||
if (_showingSavedGifs) {
|
if (_showingSavedGifs) {
|
||||||
clearInlineRows();
|
clearInlineRows(false);
|
||||||
if (_showingInlineItems) {
|
if (_showingInlineItems) {
|
||||||
const SavedGifs &saved(cSavedGifs());
|
const SavedGifs &saved(cSavedGifs());
|
||||||
if (saved.isEmpty()) {
|
if (saved.isEmpty()) {
|
||||||
|
@ -1657,15 +1707,23 @@ void StickerPanInner::refreshSavedGifs() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void StickerPanInner::inlineBotChanged() {
|
void StickerPanInner::inlineBotChanged() {
|
||||||
refreshInlineRows(0, InlineResults());
|
refreshInlineRows(0, InlineResults(), true);
|
||||||
deleteUnusedInlineLayouts();
|
deleteUnusedInlineLayouts();
|
||||||
}
|
}
|
||||||
|
|
||||||
void StickerPanInner::clearInlineRows() {
|
void StickerPanInner::clearInlineRows(bool resultsDeleted) {
|
||||||
clearSelection(true);
|
if (resultsDeleted) {
|
||||||
for (InlineRows::const_iterator i = _inlineRows.cbegin(), e = _inlineRows.cend(); i != e; ++i) {
|
if (_showingInlineItems) {
|
||||||
for (InlineItems::const_iterator j = i->items.cbegin(), end = i->items.cend(); j != end; ++j) {
|
_selected = _pressedSel = -1;
|
||||||
(*j)->setPosition(-1);
|
}
|
||||||
|
} else {
|
||||||
|
if (_showingInlineItems) {
|
||||||
|
clearSelection(true);
|
||||||
|
}
|
||||||
|
for (InlineRows::const_iterator i = _inlineRows.cbegin(), e = _inlineRows.cend(); i != e; ++i) {
|
||||||
|
for (InlineItems::const_iterator j = i->items.cbegin(), end = i->items.cend(); j != end; ++j) {
|
||||||
|
(*j)->setPosition(-1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_inlineRows.clear();
|
_inlineRows.clear();
|
||||||
|
@ -1690,9 +1748,11 @@ LayoutInlineItem *StickerPanInner::layoutPrepareInlineResult(InlineResult *resul
|
||||||
if (result->type == qstr("gif")) {
|
if (result->type == qstr("gif")) {
|
||||||
layout = new LayoutInlineGif(result, 0, false);
|
layout = new LayoutInlineGif(result, 0, false);
|
||||||
} else if (result->type == qstr("photo")) {
|
} else if (result->type == qstr("photo")) {
|
||||||
|
layout = new LayoutInlinePhoto(result, 0);
|
||||||
} else if (result->type == qstr("web_player_video")) {
|
} else if (result->type == qstr("web_player_video")) {
|
||||||
|
// layout = new LayoutInlineWebVideo(result, 0);
|
||||||
} else if (result->type == qstr("article")) {
|
} else if (result->type == qstr("article")) {
|
||||||
return 0;
|
// layout = new LayoutInlineArticle(result, 0);
|
||||||
}
|
}
|
||||||
if (!layout) return 0;
|
if (!layout) return 0;
|
||||||
|
|
||||||
|
@ -1706,7 +1766,7 @@ LayoutInlineItem *StickerPanInner::layoutPrepareInlineResult(InlineResult *resul
|
||||||
}
|
}
|
||||||
|
|
||||||
void StickerPanInner::deleteUnusedGifLayouts() {
|
void StickerPanInner::deleteUnusedGifLayouts() {
|
||||||
if (_inlineRows.isEmpty()) { // delete all
|
if (_inlineRows.isEmpty() || !_showingSavedGifs) { // delete all
|
||||||
for (GifLayouts::const_iterator i = _gifLayouts.cbegin(), e = _gifLayouts.cend(); i != e; ++i) {
|
for (GifLayouts::const_iterator i = _gifLayouts.cbegin(), e = _gifLayouts.cend(); i != e; ++i) {
|
||||||
delete i.value();
|
delete i.value();
|
||||||
}
|
}
|
||||||
|
@ -1724,7 +1784,7 @@ void StickerPanInner::deleteUnusedGifLayouts() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void StickerPanInner::deleteUnusedInlineLayouts() {
|
void StickerPanInner::deleteUnusedInlineLayouts() {
|
||||||
if (_inlineRows.isEmpty()) { // delete all
|
if (_inlineRows.isEmpty() || _showingSavedGifs) { // delete all
|
||||||
for (InlineLayouts::const_iterator i = _inlineLayouts.cbegin(), e = _inlineLayouts.cend(); i != e; ++i) {
|
for (InlineLayouts::const_iterator i = _inlineLayouts.cbegin(), e = _inlineLayouts.cend(); i != e; ++i) {
|
||||||
delete i.value();
|
delete i.value();
|
||||||
}
|
}
|
||||||
|
@ -1802,10 +1862,10 @@ uint64 StickerPanInner::currentSet(int yOffset) const {
|
||||||
return _sets.isEmpty() ? RecentStickerSetId : _sets.back().id;
|
return _sets.isEmpty() ? RecentStickerSetId : _sets.back().id;
|
||||||
}
|
}
|
||||||
|
|
||||||
void StickerPanInner::refreshInlineRows(UserData *bot, const InlineResults &results) {
|
void StickerPanInner::refreshInlineRows(UserData *bot, const InlineResults &results, bool resultsDeleted) {
|
||||||
int32 count = results.size(), until = 0, untilrow = 0, untilcol = 0;
|
int32 count = results.size(), until = 0, untilrow = 0, untilcol = 0;
|
||||||
if (!count) {
|
if (!count) {
|
||||||
_inlineRows.clear();
|
clearInlineRows(resultsDeleted);
|
||||||
_showingSavedGifs = true;
|
_showingSavedGifs = true;
|
||||||
if (_showingInlineItems) {
|
if (_showingInlineItems) {
|
||||||
refreshSavedGifs();
|
refreshSavedGifs();
|
||||||
|
@ -1816,6 +1876,7 @@ void StickerPanInner::refreshInlineRows(UserData *bot, const InlineResults &resu
|
||||||
}
|
}
|
||||||
|
|
||||||
t_assert(bot != 0);
|
t_assert(bot != 0);
|
||||||
|
_inlineBot = bot;
|
||||||
_inlineBotTitle = lng_inline_bot_results(lt_inline_bot, bot->username.isEmpty() ? bot->name : ('@' + bot->username));
|
_inlineBotTitle = lng_inline_bot_results(lt_inline_bot, bot->username.isEmpty() ? bot->name : ('@' + bot->username));
|
||||||
|
|
||||||
_showingInlineItems = true;
|
_showingInlineItems = true;
|
||||||
|
@ -2483,6 +2544,8 @@ EmojiPan::EmojiPan(QWidget *parent) : TWidget(parent)
|
||||||
|
|
||||||
connect(&e_inner, SIGNAL(selected(EmojiPtr)), this, SIGNAL(emojiSelected(EmojiPtr)));
|
connect(&e_inner, SIGNAL(selected(EmojiPtr)), this, SIGNAL(emojiSelected(EmojiPtr)));
|
||||||
connect(&s_inner, SIGNAL(selected(DocumentData*)), this, SIGNAL(stickerSelected(DocumentData*)));
|
connect(&s_inner, SIGNAL(selected(DocumentData*)), this, SIGNAL(stickerSelected(DocumentData*)));
|
||||||
|
connect(&s_inner, SIGNAL(selected(PhotoData*)), this, SIGNAL(photoSelected(PhotoData*)));
|
||||||
|
connect(&s_inner, SIGNAL(selected(InlineResult*,UserData*)), this, SIGNAL(inlineResultSelected(InlineResult*,UserData*)));
|
||||||
|
|
||||||
connect(&s_switch, SIGNAL(clicked()), this, SLOT(onSwitch()));
|
connect(&s_switch, SIGNAL(clicked()), this, SLOT(onSwitch()));
|
||||||
connect(&e_switch, SIGNAL(clicked()), this, SLOT(onSwitch()));
|
connect(&e_switch, SIGNAL(clicked()), this, SLOT(onSwitch()));
|
||||||
|
@ -3404,12 +3467,18 @@ void EmojiPan::inlineResultsDone(const MTPmessages_BotResults &result) {
|
||||||
const MTPDbotInlineResult &r(v.at(i).c_botInlineResult());
|
const MTPDbotInlineResult &r(v.at(i).c_botInlineResult());
|
||||||
result->id = qs(r.vid);
|
result->id = qs(r.vid);
|
||||||
result->type = qs(r.vtype);
|
result->type = qs(r.vtype);
|
||||||
result->title = qs(r.vtitle);
|
result->title = r.has_title() ? qs(r.vtitle) : QString();
|
||||||
result->description = qs(r.vdescription);
|
result->description = r.has_description() ? qs(r.vdescription) : QString();
|
||||||
result->url = qs(r.vurl);
|
result->url = r.has_url() ? qs(r.vurl) : QString();
|
||||||
result->thumb_url = qs(r.vthumb_url);
|
result->thumb_url = r.has_thumb_url() ? qs(r.vthumb_url) : QString();
|
||||||
result->content_type = qs(r.vcontent_type);
|
result->content_type = r.has_content_type() ? qs(r.vcontent_type) : QString();
|
||||||
result->content_url = qs(r.vcontent_url);
|
result->content_url = r.has_content_url() ? qs(r.vcontent_url) : QString();
|
||||||
|
result->width = r.has_w() ? r.vw.v : 0;
|
||||||
|
result->height = r.has_h() ? r.vh.v : 0;
|
||||||
|
result->duration = r.has_duration() ? r.vduration.v : 0;
|
||||||
|
if (!result->thumb_url.isEmpty() && (result->thumb_url.startsWith(qstr("http://"), Qt::CaseInsensitive) || result->thumb_url.startsWith(qstr("https://"), Qt::CaseInsensitive))) {
|
||||||
|
result->thumb = ImagePtr(result->thumb_url);
|
||||||
|
}
|
||||||
message = &r.vsend_message;
|
message = &r.vsend_message;
|
||||||
} break;
|
} break;
|
||||||
}
|
}
|
||||||
|
@ -3460,22 +3529,19 @@ bool EmojiPan::inlineResultsFail(const RPCError &error) {
|
||||||
void EmojiPan::queryInlineBot(UserData *bot, QString query) {
|
void EmojiPan::queryInlineBot(UserData *bot, QString query) {
|
||||||
bool force = false;
|
bool force = false;
|
||||||
if (bot != _inlineBot) {
|
if (bot != _inlineBot) {
|
||||||
LOG(("Inline bot changed! to @%1").arg(bot->username));
|
|
||||||
inlineBotChanged();
|
inlineBotChanged();
|
||||||
_inlineBot = bot;
|
_inlineBot = bot;
|
||||||
force = true;
|
force = true;
|
||||||
}
|
}
|
||||||
if (_inlineRequestId) {
|
|
||||||
MTP::cancel(_inlineRequestId);
|
|
||||||
_inlineRequestId = 0;
|
|
||||||
}
|
|
||||||
if (_inlineQuery != query || force) {
|
if (_inlineQuery != query || force) {
|
||||||
|
if (_inlineRequestId) {
|
||||||
|
MTP::cancel(_inlineRequestId);
|
||||||
|
_inlineRequestId = 0;
|
||||||
|
}
|
||||||
if (_inlineCache.contains(query)) {
|
if (_inlineCache.contains(query)) {
|
||||||
LOG(("Query %1 found in cache!").arg(query));
|
|
||||||
_inlineQuery = query;
|
_inlineQuery = query;
|
||||||
showInlineRows(true);
|
showInlineRows(true);
|
||||||
} else {
|
} else {
|
||||||
LOG(("Scheduling request for %1!").arg(query));
|
|
||||||
_inlineNextQuery = query;
|
_inlineNextQuery = query;
|
||||||
_inlineRequestTimer.start(InlineBotRequestDelay);
|
_inlineRequestTimer.start(InlineBotRequestDelay);
|
||||||
}
|
}
|
||||||
|
@ -3492,7 +3558,6 @@ void EmojiPan::onInlineRequest() {
|
||||||
nextOffset = i.value()->nextOffset;
|
nextOffset = i.value()->nextOffset;
|
||||||
if (nextOffset.isEmpty()) return;
|
if (nextOffset.isEmpty()) return;
|
||||||
}
|
}
|
||||||
LOG(("Requesting %1 with offset \"%2\"!").arg(_inlineQuery).arg(nextOffset));
|
|
||||||
_inlineRequestId = MTP::send(MTPmessages_GetInlineBotResults(_inlineBot->inputUser, MTP_string(_inlineQuery), MTP_string(nextOffset)), rpcDone(&EmojiPan::inlineResultsDone), rpcFail(&EmojiPan::inlineResultsFail));
|
_inlineRequestId = MTP::send(MTPmessages_GetInlineBotResults(_inlineBot->inputUser, MTP_string(_inlineQuery), MTP_string(nextOffset)), rpcDone(&EmojiPan::inlineResultsDone), rpcFail(&EmojiPan::inlineResultsFail));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3504,24 +3569,23 @@ void EmojiPan::showInlineRows(bool newResults) {
|
||||||
_inlineNextOffset = i.value()->nextOffset;
|
_inlineNextOffset = i.value()->nextOffset;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (clear) LOG(("Clearing results!")); else LOG(("Showing results: %1").arg(i.value()->results.size()));
|
s_inner.refreshInlineRows(_inlineBot, clear ? InlineResults() : i.value()->results, false);
|
||||||
s_inner.refreshInlineRows(_inlineBot, clear ? InlineResults() : i.value()->results);
|
|
||||||
if (newResults) s_scroll.scrollToY(0);
|
if (newResults) s_scroll.scrollToY(0);
|
||||||
if (clear && !isHidden() && _stickersShown && s_inner.inlineResultsShown()) {
|
if (clear && !isHidden() && _stickersShown && s_inner.inlineResultsShown()) {
|
||||||
hideStart();
|
hideStart();
|
||||||
} else if (!clear) {
|
} else if (!clear) {
|
||||||
_hideTimer.stop();
|
_hideTimer.stop();
|
||||||
if (!_stickersShown) {
|
if (!isHidden() || _hiding) {
|
||||||
if (!isHidden() || _hiding) {
|
if (!_stickersShown) {
|
||||||
onSwitch();
|
onSwitch();
|
||||||
} else {
|
}
|
||||||
_stickersShown = true;
|
} else {
|
||||||
if (isHidden()) {
|
_stickersShown = true;
|
||||||
show();
|
if (isHidden()) {
|
||||||
a_opacity = anim::fvalue(0, 1);
|
show();
|
||||||
a_opacity.update(0, anim::linear);
|
a_opacity = anim::fvalue(0, 1);
|
||||||
_cache = _fromCache = _toCache = QPixmap();
|
a_opacity.update(0, anim::linear);
|
||||||
}
|
_cache = _fromCache = _toCache = QPixmap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (isHidden() || _hiding) {
|
if (isHidden() || _hiding) {
|
||||||
|
|
|
@ -341,7 +341,7 @@ public:
|
||||||
void refreshStickers();
|
void refreshStickers();
|
||||||
void refreshRecentStickers(bool resize = true);
|
void refreshRecentStickers(bool resize = true);
|
||||||
void refreshSavedGifs();
|
void refreshSavedGifs();
|
||||||
void refreshInlineRows(UserData *bot, const InlineResults &results);
|
void refreshInlineRows(UserData *bot, const InlineResults &results, bool resultsDeleted);
|
||||||
void refreshRecent();
|
void refreshRecent();
|
||||||
void inlineBotChanged();
|
void inlineBotChanged();
|
||||||
|
|
||||||
|
@ -363,7 +363,7 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
~StickerPanInner() {
|
~StickerPanInner() {
|
||||||
clearInlineRows();
|
clearInlineRows(true);
|
||||||
deleteUnusedGifLayouts();
|
deleteUnusedGifLayouts();
|
||||||
deleteUnusedInlineLayouts();
|
deleteUnusedInlineLayouts();
|
||||||
}
|
}
|
||||||
|
@ -378,6 +378,9 @@ public slots:
|
||||||
signals:
|
signals:
|
||||||
|
|
||||||
void selected(DocumentData *sticker);
|
void selected(DocumentData *sticker);
|
||||||
|
void selected(PhotoData *photo);
|
||||||
|
void selected(InlineResult *result, UserData *bot);
|
||||||
|
|
||||||
void removing(quint64 setId);
|
void removing(quint64 setId);
|
||||||
|
|
||||||
void refreshIcons();
|
void refreshIcons();
|
||||||
|
@ -423,6 +426,7 @@ private:
|
||||||
QList<bool> _custom;
|
QList<bool> _custom;
|
||||||
|
|
||||||
bool _showingSavedGifs, _showingInlineItems;
|
bool _showingSavedGifs, _showingInlineItems;
|
||||||
|
UserData *_inlineBot;
|
||||||
QString _inlineBotTitle;
|
QString _inlineBotTitle;
|
||||||
uint64 _lastScrolled;
|
uint64 _lastScrolled;
|
||||||
QTimer _updateInlineItems;
|
QTimer _updateInlineItems;
|
||||||
|
@ -436,7 +440,7 @@ private:
|
||||||
};
|
};
|
||||||
typedef QVector<InlineRow> InlineRows;
|
typedef QVector<InlineRow> InlineRows;
|
||||||
InlineRows _inlineRows;
|
InlineRows _inlineRows;
|
||||||
void clearInlineRows();
|
void clearInlineRows(bool resultsDeleted);
|
||||||
|
|
||||||
typedef QMap<DocumentData*, LayoutInlineGif*> GifLayouts;
|
typedef QMap<DocumentData*, LayoutInlineGif*> GifLayouts;
|
||||||
GifLayouts _gifLayouts;
|
GifLayouts _gifLayouts;
|
||||||
|
@ -447,7 +451,7 @@ private:
|
||||||
LayoutInlineItem *layoutPrepareInlineResult(InlineResult *result, int32 position);
|
LayoutInlineItem *layoutPrepareInlineResult(InlineResult *result, int32 position);
|
||||||
|
|
||||||
void inlineRowsAddItem(DocumentData *savedGif, InlineResult *result, InlineRow &row, int32 &sumWidth);
|
void inlineRowsAddItem(DocumentData *savedGif, InlineResult *result, InlineRow &row, int32 &sumWidth);
|
||||||
void inlineRowFinalize(InlineRow &row, int32 &sumWidth, bool force = false);
|
bool inlineRowFinalize(InlineRow &row, int32 &sumWidth, bool force = false);
|
||||||
|
|
||||||
InlineRow &layoutInlineRow(InlineRow &row, int32 sumWidth = 0);
|
InlineRow &layoutInlineRow(InlineRow &row, int32 sumWidth = 0);
|
||||||
void deleteUnusedGifLayouts();
|
void deleteUnusedGifLayouts();
|
||||||
|
@ -600,6 +604,9 @@ signals:
|
||||||
|
|
||||||
void emojiSelected(EmojiPtr emoji);
|
void emojiSelected(EmojiPtr emoji);
|
||||||
void stickerSelected(DocumentData *sticker);
|
void stickerSelected(DocumentData *sticker);
|
||||||
|
void photoSelected(PhotoData *photo);
|
||||||
|
void inlineResultSelected(InlineResult *result, UserData *bot);
|
||||||
|
|
||||||
void updateStickers();
|
void updateStickers();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -294,6 +294,8 @@ void FlatTextarea::getMentionHashtagBotCommandStart(QString &start, UserData *&i
|
||||||
start = text.mid(inlineUsernameStart + inlineUsernameLength + 1);
|
start = text.mid(inlineUsernameStart + inlineUsernameLength + 1);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
inlineUsernameLength = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!inlineUsernameLength) {
|
if (!inlineUsernameLength) {
|
||||||
|
|
|
@ -1561,13 +1561,25 @@ HistoryItem *History::createItemForwarded(HistoryBlock *block, MsgId id, QDateTi
|
||||||
return regItem(new HistoryForwarded(this, block, id, date, from, msg));
|
return regItem(new HistoryForwarded(this, block, id, date, from, msg));
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryItem *History::createItemDocument(HistoryBlock *block, MsgId id, int32 flags, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption) {
|
HistoryItem *History::createItemDocument(HistoryBlock *block, MsgId id, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption) {
|
||||||
|
HistoryItem *result = 0;
|
||||||
|
|
||||||
|
if ((flags & MTPDmessage::flag_reply_to_msg_id) && replyTo > 0) {
|
||||||
|
result = new HistoryReply(this, block, id, flags, viaBotId, replyTo, date, from, doc, caption);
|
||||||
|
} else {
|
||||||
|
result = new HistoryMessage(this, block, id, flags, viaBotId, date, from, doc, caption);
|
||||||
|
}
|
||||||
|
|
||||||
|
return regItem(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
HistoryItem *History::createItemPhoto(HistoryBlock *block, MsgId id, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, PhotoData *photo, const QString &caption) {
|
||||||
HistoryItem *result = 0;
|
HistoryItem *result = 0;
|
||||||
|
|
||||||
if (flags & MTPDmessage::flag_reply_to_msg_id && replyTo > 0) {
|
if (flags & MTPDmessage::flag_reply_to_msg_id && replyTo > 0) {
|
||||||
result = new HistoryReply(this, block, id, flags, replyTo, date, from, doc, caption);
|
result = new HistoryReply(this, block, id, flags, viaBotId, replyTo, date, from, photo, caption);
|
||||||
} else {
|
} else {
|
||||||
result = new HistoryMessage(this, block, id, flags, date, from, doc, caption);
|
result = new HistoryMessage(this, block, id, flags, viaBotId, date, from, photo, caption);
|
||||||
}
|
}
|
||||||
|
|
||||||
return regItem(result);
|
return regItem(result);
|
||||||
|
@ -1633,7 +1645,7 @@ HistoryItem *History::addNewForwarded(MsgId id, QDateTime date, int32 from, Hist
|
||||||
return addNewItem(to, newBlock, createItemForwarded(to, id, date, from, item), true);
|
return addNewItem(to, newBlock, createItemForwarded(to, id, date, from, item), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryItem *History::addNewDocument(MsgId id, int32 flags, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption) {
|
HistoryItem *History::addNewDocument(MsgId id, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption) {
|
||||||
HistoryBlock *to = 0;
|
HistoryBlock *to = 0;
|
||||||
bool newBlock = blocks.isEmpty();
|
bool newBlock = blocks.isEmpty();
|
||||||
if (newBlock) {
|
if (newBlock) {
|
||||||
|
@ -1641,7 +1653,18 @@ HistoryItem *History::addNewDocument(MsgId id, int32 flags, MsgId replyTo, QDate
|
||||||
} else {
|
} else {
|
||||||
to = blocks.back();
|
to = blocks.back();
|
||||||
}
|
}
|
||||||
return addNewItem(to, newBlock, createItemDocument(to, id, flags, replyTo, date, from, doc, caption), true);
|
return addNewItem(to, newBlock, createItemDocument(to, id, flags, viaBotId, replyTo, date, from, doc, caption), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
HistoryItem *History::addNewPhoto(MsgId id, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, PhotoData *photo, const QString &caption) {
|
||||||
|
HistoryBlock *to = 0;
|
||||||
|
bool newBlock = blocks.isEmpty();
|
||||||
|
if (newBlock) {
|
||||||
|
to = new HistoryBlock(this);
|
||||||
|
} else {
|
||||||
|
to = blocks.back();
|
||||||
|
}
|
||||||
|
return addNewItem(to, newBlock, createItemPhoto(to, id, flags, viaBotId, replyTo, date, from, photo, caption), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void History::createInitialDateBlock(const QDateTime &date) {
|
void History::createInitialDateBlock(const QDateTime &date) {
|
||||||
|
@ -3209,8 +3232,8 @@ HistoryFileMedia::~HistoryFileMedia() {
|
||||||
deleteAndMark(_animation);
|
deleteAndMark(_animation);
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryPhoto::HistoryPhoto(const MTPDphoto &photo, const QString &caption, HistoryItem *parent) : HistoryFileMedia()
|
HistoryPhoto::HistoryPhoto(PhotoData *photo, const QString &caption, const HistoryItem *parent) : HistoryFileMedia()
|
||||||
, _data(App::feedPhoto(photo))
|
, _data(photo)
|
||||||
, _pixw(1)
|
, _pixw(1)
|
||||||
, _pixh(1)
|
, _pixh(1)
|
||||||
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
|
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
|
||||||
|
@ -3222,15 +3245,6 @@ HistoryPhoto::HistoryPhoto(const MTPDphoto &photo, const QString &caption, Histo
|
||||||
init();
|
init();
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryPhoto::HistoryPhoto(PhotoData *photo) : HistoryFileMedia()
|
|
||||||
, _data(photo)
|
|
||||||
, _pixw(1)
|
|
||||||
, _pixh(1) {
|
|
||||||
setLinks(new PhotoLink(_data), new PhotoSaveLink(_data), new PhotoCancelLink(_data));
|
|
||||||
|
|
||||||
init();
|
|
||||||
}
|
|
||||||
|
|
||||||
HistoryPhoto::HistoryPhoto(PeerData *chat, const MTPDphoto &photo, int32 width) : HistoryFileMedia()
|
HistoryPhoto::HistoryPhoto(PeerData *chat, const MTPDphoto &photo, int32 width) : HistoryFileMedia()
|
||||||
, _data(App::feedPhoto(photo))
|
, _data(App::feedPhoto(photo))
|
||||||
, _pixw(1)
|
, _pixw(1)
|
||||||
|
@ -5181,7 +5195,7 @@ void HistoryWebPage::initDimensions(const HistoryItem *parent) {
|
||||||
_attach = new HistoryDocument(_data->doc, QString(), parent);
|
_attach = new HistoryDocument(_data->doc, QString(), parent);
|
||||||
}
|
}
|
||||||
} else if (_data->photo) {
|
} else if (_data->photo) {
|
||||||
_attach = new HistoryPhoto(_data->photo);
|
_attach = new HistoryPhoto(_data->photo, QString(), parent);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5798,23 +5812,23 @@ HistoryMessage::HistoryMessage(History *history, HistoryBlock *block, const MTPD
|
||||||
, _text(st::msgMinWidth)
|
, _text(st::msgMinWidth)
|
||||||
, _textWidth(0)
|
, _textWidth(0)
|
||||||
, _textHeight(0)
|
, _textHeight(0)
|
||||||
|
, _viaBot(msg.has_via_bot_id() ? App::userLoaded(peerFromUser(msg.vvia_bot_id)) : 0)
|
||||||
, _media(0)
|
, _media(0)
|
||||||
, _views(msg.has_views() ? msg.vviews.v : -1)
|
, _views(msg.has_views() ? msg.vviews.v : -1) {
|
||||||
{
|
|
||||||
QString text(textClean(qs(msg.vmessage)));
|
QString text(textClean(qs(msg.vmessage)));
|
||||||
initTime();
|
initTime();
|
||||||
initMedia(msg.has_media() ? (&msg.vmedia) : 0, text);
|
initMedia(msg.has_media() ? (&msg.vmedia) : 0, text);
|
||||||
setText(text, msg.has_entities() ? entitiesFromMTP(msg.ventities.c_vector().v) : EntitiesInText());
|
setText(text, msg.has_entities() ? entitiesFromMTP(msg.ventities.c_vector().v) : EntitiesInText());
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryMessage::HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, QDateTime date, int32 from, const QString &msg, const EntitiesInText &entities, HistoryMedia *fromMedia) :
|
HistoryMessage::HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, QDateTime date, int32 from, const QString &msg, const EntitiesInText &entities, HistoryMedia *fromMedia) :
|
||||||
HistoryItem(history, block, msgId, flags, date, from)
|
HistoryItem(history, block, msgId, flags, date, from)
|
||||||
, _text(st::msgMinWidth)
|
, _text(st::msgMinWidth)
|
||||||
, _textWidth(0)
|
, _textWidth(0)
|
||||||
, _textHeight(0)
|
, _textHeight(0)
|
||||||
|
, _viaBot(viaBotId ? App::userLoaded(peerFromUser(viaBotId)) : 0)
|
||||||
, _media(0)
|
, _media(0)
|
||||||
, _views(fromChannel() ? 1 : -1)
|
, _views(fromChannel() ? 1 : -1) {
|
||||||
{
|
|
||||||
initTime();
|
initTime();
|
||||||
if (fromMedia) {
|
if (fromMedia) {
|
||||||
_media = fromMedia->clone();
|
_media = fromMedia->clone();
|
||||||
|
@ -5823,19 +5837,33 @@ HistoryItem(history, block, msgId, flags, date, from)
|
||||||
setText(msg, entities);
|
setText(msg, entities);
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryMessage::HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, QDateTime date, int32 from, DocumentData *doc, const QString &caption) :
|
HistoryMessage::HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption) :
|
||||||
HistoryItem(history, block, msgId, flags, date, from)
|
HistoryItem(history, block, msgId, flags, date, from)
|
||||||
, _text(st::msgMinWidth)
|
, _text(st::msgMinWidth)
|
||||||
, _textWidth(0)
|
, _textWidth(0)
|
||||||
, _textHeight(0)
|
, _textHeight(0)
|
||||||
|
, _viaBot(viaBotId ? App::userLoaded(peerFromUser(viaBotId)) : 0)
|
||||||
, _media(0)
|
, _media(0)
|
||||||
, _views(fromChannel() ? 1 : -1)
|
, _views(fromChannel() ? 1 : -1) {
|
||||||
{
|
|
||||||
initTime();
|
initTime();
|
||||||
initMediaFromDocument(doc, caption);
|
initMediaFromDocument(doc, caption);
|
||||||
setText(QString(), EntitiesInText());
|
setText(QString(), EntitiesInText());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
HistoryMessage::HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption) :
|
||||||
|
HistoryItem(history, block, msgId, flags, date, from)
|
||||||
|
, _text(st::msgMinWidth)
|
||||||
|
, _textWidth(0)
|
||||||
|
, _textHeight(0)
|
||||||
|
, _viaBot(viaBotId ? App::userLoaded(peerFromUser(viaBotId)) : 0)
|
||||||
|
, _media(0)
|
||||||
|
, _views(fromChannel() ? 1 : -1) {
|
||||||
|
initTime();
|
||||||
|
_media = new HistoryPhoto(photo, caption, this);
|
||||||
|
_media->regItem(this);
|
||||||
|
setText(QString(), EntitiesInText());
|
||||||
|
}
|
||||||
|
|
||||||
QString formatViewsCount(int32 views) {
|
QString formatViewsCount(int32 views) {
|
||||||
if (views > 999999) {
|
if (views > 999999) {
|
||||||
views /= 100000;
|
views /= 100000;
|
||||||
|
@ -5886,7 +5914,7 @@ void HistoryMessage::initMedia(const MTPMessageMedia *media, QString ¤tTex
|
||||||
case mtpc_messageMediaPhoto: {
|
case mtpc_messageMediaPhoto: {
|
||||||
const MTPDmessageMediaPhoto &photo(media->c_messageMediaPhoto());
|
const MTPDmessageMediaPhoto &photo(media->c_messageMediaPhoto());
|
||||||
if (photo.vphoto.type() == mtpc_photo) {
|
if (photo.vphoto.type() == mtpc_photo) {
|
||||||
_media = new HistoryPhoto(photo.vphoto.c_photo(), qs(photo.vcaption), this);
|
_media = new HistoryPhoto(App::feedPhoto(photo.vphoto.c_photo()), qs(photo.vcaption), this);
|
||||||
}
|
}
|
||||||
} break;
|
} break;
|
||||||
case mtpc_messageMediaVideo: {
|
case mtpc_messageMediaVideo: {
|
||||||
|
@ -6468,7 +6496,8 @@ HistoryMessage::~HistoryMessage() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryForwarded::HistoryForwarded(History *history, HistoryBlock *block, const MTPDmessage &msg) : HistoryMessage(history, block, msg)
|
HistoryForwarded::HistoryForwarded(History *history, HistoryBlock *block, const MTPDmessage &msg)
|
||||||
|
: HistoryMessage(history, block, msg)
|
||||||
, fwdDate(::date(msg.vfwd_date))
|
, fwdDate(::date(msg.vfwd_date))
|
||||||
, fwdFrom(App::peer(peerFromMTP(msg.vfwd_from_id)))
|
, fwdFrom(App::peer(peerFromMTP(msg.vfwd_from_id)))
|
||||||
, fwdFromVersion(fwdFrom->nameVersion)
|
, fwdFromVersion(fwdFrom->nameVersion)
|
||||||
|
@ -6476,7 +6505,8 @@ HistoryForwarded::HistoryForwarded(History *history, HistoryBlock *block, const
|
||||||
fwdNameUpdated();
|
fwdNameUpdated();
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryForwarded::HistoryForwarded(History *history, HistoryBlock *block, MsgId id, QDateTime date, int32 from, HistoryMessage *msg) : HistoryMessage(history, block, id, newMessageFlags(history->peer) | (!history->peer->isChannel() && msg->getMedia() && (msg->getMedia()->type() == MediaTypeAudio/* || msg->getMedia()->type() == MediaTypeVideo*/) ? MTPDmessage::flag_media_unread : 0), date, from, msg->HistoryMessage::originalText(), msg->HistoryMessage::originalEntities(), msg->getMedia())
|
HistoryForwarded::HistoryForwarded(History *history, HistoryBlock *block, MsgId id, QDateTime date, int32 from, HistoryMessage *msg)
|
||||||
|
: HistoryMessage(history, block, id, newMessageFlags(history->peer) | (!history->peer->isChannel() && msg->getMedia() && (msg->getMedia()->type() == MediaTypeAudio/* || msg->getMedia()->type() == MediaTypeVideo*/) ? MTPDmessage::flag_media_unread : 0), msg->viaBot() ? peerToUser(msg->viaBot()->id) : 0, date, from, msg->HistoryMessage::originalText(), msg->HistoryMessage::originalEntities(), msg->getMedia())
|
||||||
, fwdDate(msg->dateForwarded())
|
, fwdDate(msg->dateForwarded())
|
||||||
, fwdFrom(msg->fromForwarded())
|
, fwdFrom(msg->fromForwarded())
|
||||||
, fwdFromVersion(fwdFrom->nameVersion)
|
, fwdFromVersion(fwdFrom->nameVersion)
|
||||||
|
@ -6647,8 +6677,8 @@ HistoryReply::HistoryReply(History *history, HistoryBlock *block, const MTPDmess
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
HistoryReply::HistoryReply(History *history, HistoryBlock *block, MsgId msgId, int32 flags, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption)
|
HistoryReply::HistoryReply(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption)
|
||||||
: HistoryMessage(history, block, msgId, flags, date, from, doc, caption)
|
: HistoryMessage(history, block, msgId, flags, viaBotId, date, from, doc, caption)
|
||||||
, replyToMsgId(replyTo)
|
, replyToMsgId(replyTo)
|
||||||
, replyToMsg(0)
|
, replyToMsg(0)
|
||||||
, replyToVersion(0)
|
, replyToVersion(0)
|
||||||
|
@ -6658,6 +6688,16 @@ HistoryReply::HistoryReply(History *history, HistoryBlock *block, MsgId msgId, i
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
HistoryReply::HistoryReply(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, PhotoData *photo, const QString &caption)
|
||||||
|
: HistoryMessage(history, block, msgId, flags, viaBotId, date, from, photo, caption)
|
||||||
|
, replyToMsgId(replyTo)
|
||||||
|
, replyToMsg(0)
|
||||||
|
, replyToVersion(0)
|
||||||
|
, _maxReplyWidth(0) {
|
||||||
|
if (!updateReplyTo() && App::api()) {
|
||||||
|
App::api()->requestReplyTo(this, history->peer->asChannel(), replyToMsgId);
|
||||||
|
}
|
||||||
|
}
|
||||||
QString HistoryReply::selectedText(uint32 selection) const {
|
QString HistoryReply::selectedText(uint32 selection) const {
|
||||||
if (selection != FullSelection || !replyToMsg) return HistoryMessage::selectedText(selection);
|
if (selection != FullSelection || !replyToMsg) return HistoryMessage::selectedText(selection);
|
||||||
QString result, original = HistoryMessage::selectedText(selection);
|
QString result, original = HistoryMessage::selectedText(selection);
|
||||||
|
|
|
@ -195,13 +195,15 @@ public:
|
||||||
|
|
||||||
HistoryItem *createItem(HistoryBlock *block, const MTPMessage &msg, bool applyServiceAction);
|
HistoryItem *createItem(HistoryBlock *block, const MTPMessage &msg, bool applyServiceAction);
|
||||||
HistoryItem *createItemForwarded(HistoryBlock *block, MsgId id, QDateTime date, int32 from, HistoryMessage *msg);
|
HistoryItem *createItemForwarded(HistoryBlock *block, MsgId id, QDateTime date, int32 from, HistoryMessage *msg);
|
||||||
HistoryItem *createItemDocument(HistoryBlock *block, MsgId id, int32 flags, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption);
|
HistoryItem *createItemDocument(HistoryBlock *block, MsgId id, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption);
|
||||||
|
HistoryItem *createItemPhoto(HistoryBlock *block, MsgId id, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, PhotoData *photo, const QString &caption);
|
||||||
|
|
||||||
HistoryItem *addNewService(MsgId msgId, QDateTime date, const QString &text, int32 flags = 0, HistoryMedia *media = 0, bool newMsg = true);
|
HistoryItem *addNewService(MsgId msgId, QDateTime date, const QString &text, int32 flags = 0, HistoryMedia *media = 0, bool newMsg = true);
|
||||||
HistoryItem *addNewMessage(const MTPMessage &msg, NewMessageType type);
|
HistoryItem *addNewMessage(const MTPMessage &msg, NewMessageType type);
|
||||||
HistoryItem *addToHistory(const MTPMessage &msg);
|
HistoryItem *addToHistory(const MTPMessage &msg);
|
||||||
HistoryItem *addNewForwarded(MsgId id, QDateTime date, int32 from, HistoryMessage *item);
|
HistoryItem *addNewForwarded(MsgId id, QDateTime date, int32 from, HistoryMessage *item);
|
||||||
HistoryItem *addNewDocument(MsgId id, int32 flags, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption);
|
HistoryItem *addNewDocument(MsgId id, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption);
|
||||||
|
HistoryItem *addNewPhoto(MsgId id, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, PhotoData *photo, const QString &caption);
|
||||||
|
|
||||||
void addOlderSlice(const QVector<MTPMessage> &slice, const QVector<MTPMessageGroup> *collapsed);
|
void addOlderSlice(const QVector<MTPMessage> &slice, const QVector<MTPMessageGroup> *collapsed);
|
||||||
void addNewerSlice(const QVector<MTPMessage> &slice, const QVector<MTPMessageGroup> *collapsed);
|
void addNewerSlice(const QVector<MTPMessage> &slice, const QVector<MTPMessageGroup> *collapsed);
|
||||||
|
@ -1271,8 +1273,7 @@ private:
|
||||||
class HistoryPhoto : public HistoryFileMedia {
|
class HistoryPhoto : public HistoryFileMedia {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
HistoryPhoto(const MTPDphoto &photo, const QString &caption, HistoryItem *parent);
|
HistoryPhoto(PhotoData *photo, const QString &caption, const HistoryItem *parent);
|
||||||
HistoryPhoto(PhotoData *photo);
|
|
||||||
HistoryPhoto(PeerData *chat, const MTPDphoto &photo, int32 width = 0);
|
HistoryPhoto(PeerData *chat, const MTPDphoto &photo, int32 width = 0);
|
||||||
HistoryPhoto(const HistoryPhoto &other);
|
HistoryPhoto(const HistoryPhoto &other);
|
||||||
void init();
|
void init();
|
||||||
|
@ -1904,8 +1905,9 @@ class HistoryMessage : public HistoryItem {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
HistoryMessage(History *history, HistoryBlock *block, const MTPDmessage &msg);
|
HistoryMessage(History *history, HistoryBlock *block, const MTPDmessage &msg);
|
||||||
HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, QDateTime date, int32 from, const QString &msg, const EntitiesInText &entities, HistoryMedia *media); // local forwarded
|
HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, QDateTime date, int32 from, const QString &msg, const EntitiesInText &entities, HistoryMedia *media); // local forwarded
|
||||||
HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, QDateTime date, int32 from, DocumentData *doc, const QString &caption); // local sticker and reply sticker
|
HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption); // local document
|
||||||
|
HistoryMessage(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption); // local photo
|
||||||
|
|
||||||
void initTime();
|
void initTime();
|
||||||
void initMedia(const MTPMessageMedia *media, QString ¤tText);
|
void initMedia(const MTPMessageMedia *media, QString ¤tText);
|
||||||
|
@ -1913,6 +1915,10 @@ public:
|
||||||
void initDimensions();
|
void initDimensions();
|
||||||
void fromNameUpdated() const;
|
void fromNameUpdated() const;
|
||||||
|
|
||||||
|
UserData *viaBot() const {
|
||||||
|
return _viaBot;
|
||||||
|
}
|
||||||
|
|
||||||
int32 plainMaxWidth() const;
|
int32 plainMaxWidth() const;
|
||||||
void countPositionAndSize(int32 &left, int32 &width) const;
|
void countPositionAndSize(int32 &left, int32 &width) const;
|
||||||
|
|
||||||
|
@ -2030,6 +2036,7 @@ protected:
|
||||||
Text _text;
|
Text _text;
|
||||||
|
|
||||||
int32 _textWidth, _textHeight;
|
int32 _textWidth, _textHeight;
|
||||||
|
UserData *_viaBot;
|
||||||
|
|
||||||
HistoryMedia *_media;
|
HistoryMedia *_media;
|
||||||
QString _timeText;
|
QString _timeText;
|
||||||
|
@ -2091,7 +2098,8 @@ class HistoryReply : public HistoryMessage {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
HistoryReply(History *history, HistoryBlock *block, const MTPDmessage &msg);
|
HistoryReply(History *history, HistoryBlock *block, const MTPDmessage &msg);
|
||||||
HistoryReply(History *history, HistoryBlock *block, MsgId msgId, int32 flags, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption);
|
HistoryReply(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, DocumentData *doc, const QString &caption);
|
||||||
|
HistoryReply(History *history, HistoryBlock *block, MsgId msgId, int32 flags, int32 viaBotId, MsgId replyTo, QDateTime date, int32 from, PhotoData *photo, const QString &caption);
|
||||||
|
|
||||||
void initDimensions();
|
void initDimensions();
|
||||||
|
|
||||||
|
|
|
@ -2708,6 +2708,8 @@ HistoryWidget::HistoryWidget(QWidget *parent) : TWidget(parent)
|
||||||
connect(&_scrollTimer, SIGNAL(timeout()), this, SLOT(onScrollTimer()));
|
connect(&_scrollTimer, SIGNAL(timeout()), this, SLOT(onScrollTimer()));
|
||||||
connect(&_emojiPan, SIGNAL(emojiSelected(EmojiPtr)), &_field, SLOT(onEmojiInsert(EmojiPtr)));
|
connect(&_emojiPan, SIGNAL(emojiSelected(EmojiPtr)), &_field, SLOT(onEmojiInsert(EmojiPtr)));
|
||||||
connect(&_emojiPan, SIGNAL(stickerSelected(DocumentData*)), this, SLOT(onStickerSend(DocumentData*)));
|
connect(&_emojiPan, SIGNAL(stickerSelected(DocumentData*)), this, SLOT(onStickerSend(DocumentData*)));
|
||||||
|
connect(&_emojiPan, SIGNAL(photoSelected(PhotoData*)), this, SLOT(onPhotoSend(PhotoData*)));
|
||||||
|
connect(&_emojiPan, SIGNAL(inlineResultSelected(InlineResult*,UserData*)), this, SLOT(onInlineResultSend(InlineResult*,UserData*)));
|
||||||
connect(&_emojiPan, SIGNAL(updateStickers()), this, SLOT(updateStickers()));
|
connect(&_emojiPan, SIGNAL(updateStickers()), this, SLOT(updateStickers()));
|
||||||
connect(&_sendActionStopTimer, SIGNAL(timeout()), this, SLOT(onCancelSendAction()));
|
connect(&_sendActionStopTimer, SIGNAL(timeout()), this, SLOT(onCancelSendAction()));
|
||||||
connect(&_previewTimer, SIGNAL(timeout()), this, SLOT(onPreviewTimeout()));
|
connect(&_previewTimer, SIGNAL(timeout()), this, SLOT(onPreviewTimeout()));
|
||||||
|
@ -6267,7 +6269,15 @@ void HistoryWidget::onFieldTabbed() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryWidget::onStickerSend(DocumentData *sticker) {
|
void HistoryWidget::onStickerSend(DocumentData *sticker) {
|
||||||
if (!_history || !sticker || !canSendMessages(_peer)) return;
|
sendExistingDocument(sticker, QString(), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryWidget::onPhotoSend(PhotoData *photo) {
|
||||||
|
sendExistingPhoto(photo, QString(), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryWidget::onInlineResultSend(InlineResult *result, UserData *bot) {
|
||||||
|
if (!_history || !result || !canSendMessages(_peer)) return;
|
||||||
|
|
||||||
App::main()->readServerHistory(_history, false);
|
App::main()->readServerHistory(_history, false);
|
||||||
fastShowAtEnd(_history);
|
fastShowAtEnd(_history);
|
||||||
|
@ -6290,13 +6300,156 @@ void HistoryWidget::onStickerSend(DocumentData *sticker) {
|
||||||
} else {
|
} else {
|
||||||
flags |= MTPDmessage::flag_from_id;
|
flags |= MTPDmessage::flag_from_id;
|
||||||
}
|
}
|
||||||
_history->addNewDocument(newId.msg, flags, replyToId(), date(MTP_int(unixtime())), fromChannelName ? 0 : MTP::authedId(), sticker, QString());
|
if (bot) {
|
||||||
|
flags |= MTPDmessage::flag_via_bot_id;
|
||||||
|
}
|
||||||
|
|
||||||
_history->sendRequestId = MTP::send(MTPmessages_SendMedia(MTP_int(sendFlags), _peer->input, MTP_int(replyToId()), MTP_inputMediaDocument(MTP_inputDocument(MTP_long(sticker->id), MTP_long(sticker->access)), MTP_string("")), MTP_long(randomId), MTPnullMarkup), App::main()->rpcDone(&MainWidget::sentUpdatesReceived), App::main()->rpcFail(&MainWidget::sendMessageFail), 0, 0, _history->sendRequestId);
|
if (result->message.isEmpty()) {
|
||||||
|
if (result->doc) {
|
||||||
|
_history->addNewDocument(newId.msg, flags, bot ? peerToUser(bot->id) : 0, replyToId(), date(MTP_int(unixtime())), fromChannelName ? 0 : MTP::authedId(), result->doc, result->caption);
|
||||||
|
} else if (result->photo) {
|
||||||
|
_history->addNewPhoto(newId.msg, flags, bot ? peerToUser(bot->id) : 0, replyToId(), date(MTP_int(unixtime())), fromChannelName ? 0 : MTP::authedId(), result->photo, result->caption);
|
||||||
|
} else if (result->type == qstr("gif")) {
|
||||||
|
MTPPhotoSize thumbSize;
|
||||||
|
QPixmap thumb;
|
||||||
|
int32 tw = result->thumb->width(), th = result->thumb->height();
|
||||||
|
if (tw > 0 && th > 0 && tw < 20 * th && th < 20 * tw && result->thumb->loaded()) {
|
||||||
|
if (tw > th) {
|
||||||
|
if (tw > 90) {
|
||||||
|
th = th * 90 / tw;
|
||||||
|
tw = 90;
|
||||||
|
}
|
||||||
|
} else if (th > 90) {
|
||||||
|
tw = tw * 90 / th;
|
||||||
|
th = 90;
|
||||||
|
}
|
||||||
|
thumbSize = MTP_photoSize(MTP_string(""), MTP_fileLocationUnavailable(MTP_long(0), MTP_int(0), MTP_long(0)), MTP_int(tw), MTP_int(th), MTP_int(0));
|
||||||
|
thumb = result->thumb->pixNoCache(tw, th, true, false, false);
|
||||||
|
} else {
|
||||||
|
tw = th = 0;
|
||||||
|
thumbSize = MTP_photoSizeEmpty(MTP_string(""));
|
||||||
|
}
|
||||||
|
uint64 docId = MTP::nonce<uint64>();
|
||||||
|
QVector<MTPDocumentAttribute> attributes(1, MTP_documentAttributeFilename(MTP_string((result->content_type == qstr("video/mp4") ? "animation.gif.mp4" : "animation.gif"))));
|
||||||
|
attributes.push_back(MTP_documentAttributeAnimated());
|
||||||
|
attributes.push_back(MTP_documentAttributeVideo(MTP_int(result->duration), MTP_int(result->width), MTP_int(result->height)));
|
||||||
|
MTPDocument document = MTP_document(MTP_long(docId), MTP_long(0), MTP_int(unixtime()), MTP_string(result->content_type), MTP_int(result->data().size()), thumbSize, MTP_int(MTP::maindc()), MTP_vector<MTPDocumentAttribute>(attributes));
|
||||||
|
if (tw > 0 && th > 0) {
|
||||||
|
App::feedDocument(document, thumb);
|
||||||
|
}
|
||||||
|
Local::writeStickerImage(mediaKey(DocumentFileLocation, MTP::maindc(), docId), result->data());
|
||||||
|
_history->addNewMessage(MTP_message(MTP_int(flags), MTP_int(newId.msg), MTP_int(fromChannelName ? 0 : MTP::authedId()), peerToMTP(_history->peer->id), MTPPeer(), MTPint(), MTP_int(bot ? peerToUser(bot->id) : 0), MTP_int(replyToId()), MTP_int(unixtime()), MTP_string(""), MTP_messageMediaDocument(document, MTP_string(result->caption)), MTPnullMarkup, MTPnullEntities, MTP_int(1)), NewMessageUnread);
|
||||||
|
} else if (result->type == qstr("photo")) {
|
||||||
|
QImage fileThumb(result->thumb->pix().toImage());
|
||||||
|
|
||||||
|
PreparedPhotoThumbs photoThumbs;
|
||||||
|
QVector<MTPPhotoSize> photoSizes;
|
||||||
|
|
||||||
|
QPixmap thumb = (fileThumb.width() > 100 || fileThumb.height() > 100) ? QPixmap::fromImage(fileThumb.scaled(100, 100, Qt::KeepAspectRatio, Qt::SmoothTransformation), Qt::ColorOnly) : QPixmap::fromImage(fileThumb);
|
||||||
|
photoThumbs.insert('s', thumb);
|
||||||
|
photoSizes.push_back(MTP_photoSize(MTP_string("s"), MTP_fileLocationUnavailable(MTP_long(0), MTP_int(0), MTP_long(0)), MTP_int(thumb.width()), MTP_int(thumb.height()), MTP_int(0)));
|
||||||
|
|
||||||
|
QPixmap medium = (fileThumb.width() > 320 || fileThumb.height() > 320) ? QPixmap::fromImage(fileThumb.scaled(320, 320, Qt::KeepAspectRatio, Qt::SmoothTransformation), Qt::ColorOnly) : QPixmap::fromImage(fileThumb);
|
||||||
|
photoThumbs.insert('m', medium);
|
||||||
|
photoSizes.push_back(MTP_photoSize(MTP_string("m"), MTP_fileLocationUnavailable(MTP_long(0), MTP_int(0), MTP_long(0)), MTP_int(medium.width()), MTP_int(medium.height()), MTP_int(0)));
|
||||||
|
|
||||||
|
MTPPhoto photo = MTP_photo(MTP_long(MTP::nonce<uint64>()), MTP_long(0), MTP_int(unixtime()), MTP_vector<MTPPhotoSize>(photoSizes));
|
||||||
|
_history->addNewMessage(MTP_message(MTP_int(flags), MTP_int(newId.msg), MTP_int(fromChannelName ? 0 : MTP::authedId()), peerToMTP(_history->peer->id), MTPPeer(), MTPint(), MTP_int(bot ? peerToUser(bot->id) : 0), MTP_int(replyToId()), MTP_int(unixtime()), MTP_string(""), MTP_messageMediaPhoto(photo, MTP_string(result->caption)), MTPnullMarkup, MTPnullEntities, MTP_int(1)), NewMessageUnread);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (result->noWebPage) {
|
||||||
|
sendFlags |= MTPmessages_SendMessage::flag_no_webpage;
|
||||||
|
}
|
||||||
|
_history->addNewMessage(MTP_message(MTP_int(flags), MTP_int(newId.msg), MTP_int(fromChannelName ? 0 : MTP::authedId()), peerToMTP(_history->peer->id), MTPPeer(), MTPint(), MTP_int(bot ? peerToUser(bot->id) : 0), MTP_int(replyToId()), MTP_int(unixtime()), MTP_string(result->message), MTP_messageMediaEmpty(), MTPnullMarkup, linksToMTP(result->entities), MTP_int(1)), NewMessageUnread);
|
||||||
|
}
|
||||||
|
_history->sendRequestId = MTP::send(MTPmessages_SendInlineBotResult(MTP_int(sendFlags), _peer->input, MTP_int(replyToId()), MTP_long(randomId), MTP_long(result->queryId), MTP_string(result->id)), App::main()->rpcDone(&MainWidget::sentUpdatesReceived), App::main()->rpcFail(&MainWidget::sendMessageFail), 0, 0, _history->sendRequestId);
|
||||||
App::main()->finishForwarding(_history, _broadcast.checked());
|
App::main()->finishForwarding(_history, _broadcast.checked());
|
||||||
cancelReply(lastKeyboardUsed);
|
cancelReply(lastKeyboardUsed);
|
||||||
|
|
||||||
if (sticker->sticker()) App::main()->incrementSticker(sticker);
|
App::historyRegRandom(randomId, newId);
|
||||||
|
|
||||||
|
setFieldText(QString());
|
||||||
|
_saveDraftText = true;
|
||||||
|
_saveDraftStart = getms();
|
||||||
|
onDraftSave();
|
||||||
|
|
||||||
|
if (!_attachMention.isHidden()) _attachMention.hideStart();
|
||||||
|
if (!_attachType.isHidden()) _attachType.hideStart();
|
||||||
|
if (!_emojiPan.isHidden()) _emojiPan.hideStart();
|
||||||
|
|
||||||
|
_field.setFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryWidget::sendExistingDocument(DocumentData *doc, const QString &caption, UserData *bot) {
|
||||||
|
if (!_history || !doc || !canSendMessages(_peer)) return;
|
||||||
|
|
||||||
|
App::main()->readServerHistory(_history, false);
|
||||||
|
fastShowAtEnd(_history);
|
||||||
|
|
||||||
|
uint64 randomId = MTP::nonce<uint64>();
|
||||||
|
FullMsgId newId(_channel, clientMsgId());
|
||||||
|
|
||||||
|
bool lastKeyboardUsed = lastForceReplyReplied();
|
||||||
|
|
||||||
|
bool out = !_peer->isSelf(), unread = !_peer->isSelf();
|
||||||
|
int32 flags = newMessageFlags(_peer) | MTPDmessage::flag_media; // unread, out
|
||||||
|
int32 sendFlags = 0;
|
||||||
|
if (replyToId()) {
|
||||||
|
flags |= MTPDmessage::flag_reply_to_msg_id;
|
||||||
|
sendFlags |= MTPmessages_SendMedia::flag_reply_to_msg_id;
|
||||||
|
}
|
||||||
|
bool fromChannelName = _peer->isChannel() && !_peer->isMegagroup() && _peer->asChannel()->canPublish() && (_peer->asChannel()->isBroadcast() || _broadcast.checked());
|
||||||
|
if (fromChannelName) {
|
||||||
|
sendFlags |= MTPmessages_SendMedia::flag_broadcast;
|
||||||
|
} else {
|
||||||
|
flags |= MTPDmessage::flag_from_id;
|
||||||
|
}
|
||||||
|
_history->addNewDocument(newId.msg, flags, bot ? peerToUser(bot->id) : 0, replyToId(), date(MTP_int(unixtime())), fromChannelName ? 0 : MTP::authedId(), doc, caption);
|
||||||
|
|
||||||
|
_history->sendRequestId = MTP::send(MTPmessages_SendMedia(MTP_int(sendFlags), _peer->input, MTP_int(replyToId()), MTP_inputMediaDocument(MTP_inputDocument(MTP_long(doc->id), MTP_long(doc->access)), MTP_string(caption)), MTP_long(randomId), MTPnullMarkup), App::main()->rpcDone(&MainWidget::sentUpdatesReceived), App::main()->rpcFail(&MainWidget::sendMessageFail), 0, 0, _history->sendRequestId);
|
||||||
|
App::main()->finishForwarding(_history, _broadcast.checked());
|
||||||
|
cancelReply(lastKeyboardUsed);
|
||||||
|
|
||||||
|
if (doc->sticker()) App::main()->incrementSticker(doc);
|
||||||
|
|
||||||
|
App::historyRegRandom(randomId, newId);
|
||||||
|
|
||||||
|
if (!_attachMention.isHidden()) _attachMention.hideStart();
|
||||||
|
if (!_attachType.isHidden()) _attachType.hideStart();
|
||||||
|
if (!_emojiPan.isHidden()) _emojiPan.hideStart();
|
||||||
|
|
||||||
|
_field.setFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryWidget::sendExistingPhoto(PhotoData *photo, const QString &caption, UserData *bot) {
|
||||||
|
if (!_history || !photo || !canSendMessages(_peer)) return;
|
||||||
|
|
||||||
|
App::main()->readServerHistory(_history, false);
|
||||||
|
fastShowAtEnd(_history);
|
||||||
|
|
||||||
|
uint64 randomId = MTP::nonce<uint64>();
|
||||||
|
FullMsgId newId(_channel, clientMsgId());
|
||||||
|
|
||||||
|
bool lastKeyboardUsed = lastForceReplyReplied();
|
||||||
|
|
||||||
|
bool out = !_peer->isSelf(), unread = !_peer->isSelf();
|
||||||
|
int32 flags = newMessageFlags(_peer) | MTPDmessage::flag_media; // unread, out
|
||||||
|
int32 sendFlags = 0;
|
||||||
|
if (replyToId()) {
|
||||||
|
flags |= MTPDmessage::flag_reply_to_msg_id;
|
||||||
|
sendFlags |= MTPmessages_SendMedia::flag_reply_to_msg_id;
|
||||||
|
}
|
||||||
|
bool fromChannelName = _peer->isChannel() && !_peer->isMegagroup() && _peer->asChannel()->canPublish() && (_peer->asChannel()->isBroadcast() || _broadcast.checked());
|
||||||
|
if (fromChannelName) {
|
||||||
|
sendFlags |= MTPmessages_SendMedia::flag_broadcast;
|
||||||
|
} else {
|
||||||
|
flags |= MTPDmessage::flag_from_id;
|
||||||
|
}
|
||||||
|
_history->addNewPhoto(newId.msg, flags, bot ? peerToUser(bot->id) : 0, replyToId(), date(MTP_int(unixtime())), fromChannelName ? 0 : MTP::authedId(), photo, caption);
|
||||||
|
|
||||||
|
_history->sendRequestId = MTP::send(MTPmessages_SendMedia(MTP_int(sendFlags), _peer->input, MTP_int(replyToId()), MTP_inputMediaPhoto(MTP_inputPhoto(MTP_long(photo->id), MTP_long(photo->access)), MTP_string(caption)), MTP_long(randomId), MTPnullMarkup), App::main()->rpcDone(&MainWidget::sentUpdatesReceived), App::main()->rpcFail(&MainWidget::sendMessageFail), 0, 0, _history->sendRequestId);
|
||||||
|
App::main()->finishForwarding(_history, _broadcast.checked());
|
||||||
|
cancelReply(lastKeyboardUsed);
|
||||||
|
|
||||||
App::historyRegRandom(randomId, newId);
|
App::historyRegRandom(randomId, newId);
|
||||||
|
|
||||||
|
|
|
@ -638,6 +638,8 @@ public slots:
|
||||||
|
|
||||||
void onFieldTabbed();
|
void onFieldTabbed();
|
||||||
void onStickerSend(DocumentData *sticker);
|
void onStickerSend(DocumentData *sticker);
|
||||||
|
void onPhotoSend(PhotoData *photo);
|
||||||
|
void onInlineResultSend(InlineResult *result, UserData *bot);
|
||||||
|
|
||||||
void onVisibleChanged();
|
void onVisibleChanged();
|
||||||
|
|
||||||
|
@ -679,6 +681,9 @@ private:
|
||||||
IconedButton _replyForwardPreviewCancel;
|
IconedButton _replyForwardPreviewCancel;
|
||||||
void updateReplyToName();
|
void updateReplyToName();
|
||||||
|
|
||||||
|
void sendExistingDocument(DocumentData *doc, const QString &caption, UserData *bot);
|
||||||
|
void sendExistingPhoto(PhotoData *photo, const QString &caption, UserData *bot);
|
||||||
|
|
||||||
void drawField(Painter &p);
|
void drawField(Painter &p);
|
||||||
void drawRecordButton(Painter &p);
|
void drawRecordButton(Painter &p);
|
||||||
void drawRecording(Painter &p);
|
void drawRecording(Painter &p);
|
||||||
|
|
|
@ -1314,7 +1314,7 @@ void LayoutInlineItem::preload() {
|
||||||
_result->photo->thumb->load();
|
_result->photo->thumb->load();
|
||||||
} else if (_result->doc) {
|
} else if (_result->doc) {
|
||||||
_result->doc->thumb->load();
|
_result->doc->thumb->load();
|
||||||
} else if (!_result->thumb_url.isEmpty()) {
|
} else if (!_result->thumb->isNull()) {
|
||||||
_result->thumb->load();
|
_result->thumb->load();
|
||||||
}
|
}
|
||||||
} else if (_doc) {
|
} else if (_doc) {
|
||||||
|
@ -1324,6 +1324,12 @@ void LayoutInlineItem::preload() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void LayoutInlineItem::update() {
|
||||||
|
if (_position >= 0) {
|
||||||
|
Ui::repaintInlineItem(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
LayoutInlineGif::LayoutInlineGif(InlineResult *result, DocumentData *doc, bool saved) : LayoutInlineItem(result, doc, 0)
|
LayoutInlineGif::LayoutInlineGif(InlineResult *result, DocumentData *doc, bool saved) : LayoutInlineItem(result, doc, 0)
|
||||||
, _state(0)
|
, _state(0)
|
||||||
, _gif(0)
|
, _gif(0)
|
||||||
|
@ -1365,7 +1371,7 @@ void DeleteSavedGifLink::onClick(Qt::MouseButton button) const {
|
||||||
}
|
}
|
||||||
|
|
||||||
void LayoutInlineGif::paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const {
|
void LayoutInlineGif::paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const {
|
||||||
content_automaticLoad();
|
// content_automaticLoad();
|
||||||
|
|
||||||
bool loaded = content_loaded(), loading = content_loading(), displayLoading = content_displayLoading();
|
bool loaded = content_loaded(), loading = content_loading(), displayLoading = content_displayLoading();
|
||||||
if (loaded && !gif() && _gif != BadClipReader) {
|
if (loaded && !gif() && _gif != BadClipReader) {
|
||||||
|
@ -1394,7 +1400,11 @@ void LayoutInlineGif::paint(Painter &p, const QRect &clip, uint32 selection, con
|
||||||
p.drawPixmap(r.topLeft(), _gif->current(frame.width(), frame.height(), _width, height, ctx->paused ? 0 : context->ms));
|
p.drawPixmap(r.topLeft(), _gif->current(frame.width(), frame.height(), _width, height, ctx->paused ? 0 : context->ms));
|
||||||
} else {
|
} else {
|
||||||
prepareThumb(_width, height, frame);
|
prepareThumb(_width, height, frame);
|
||||||
p.drawPixmap(r.topLeft(), _thumb);
|
if (_thumb.isNull()) {
|
||||||
|
p.fillRect(r, st::overviewPhotoBg);
|
||||||
|
} else {
|
||||||
|
p.drawPixmap(r.topLeft(), _thumb);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (radial || (!_gif && !loaded && !loading) || (_gif == BadClipReader)) {
|
if (radial || (!_gif && !loaded && !loading) || (_gif == BadClipReader)) {
|
||||||
|
@ -1468,7 +1478,8 @@ void LayoutInlineGif::linkOver(const TextLinkPtr &link) {
|
||||||
void LayoutInlineGif::linkOut(const TextLinkPtr &link) {
|
void LayoutInlineGif::linkOut(const TextLinkPtr &link) {
|
||||||
if (_delete && link == _delete) {
|
if (_delete && link == _delete) {
|
||||||
if (_state & StateDeleteOver) {
|
if (_state & StateDeleteOver) {
|
||||||
EnsureAnimation(_a_deleteOver, 1, func(this, &LayoutInlineGif::update));
|
update();
|
||||||
|
EnsureAnimation(_a_deleteOver, 1, func(this, &LayoutInlineItem::update));
|
||||||
_state &= ~StateDeleteOver;
|
_state &= ~StateDeleteOver;
|
||||||
_a_deleteOver.start(0, st::stickersRowDuration);
|
_a_deleteOver.start(0, st::stickersRowDuration);
|
||||||
}
|
}
|
||||||
|
@ -1477,7 +1488,7 @@ void LayoutInlineGif::linkOut(const TextLinkPtr &link) {
|
||||||
if (!content_loaded()) {
|
if (!content_loaded()) {
|
||||||
ensureAnimation();
|
ensureAnimation();
|
||||||
if (_state & StateOver) {
|
if (_state & StateOver) {
|
||||||
EnsureAnimation(_animation->_a_over, 1, func(this, &LayoutInlineGif::update));
|
EnsureAnimation(_animation->_a_over, 1, func(this, &LayoutInlineItem::update));
|
||||||
_animation->_a_over.start(0, st::stickersRowDuration);
|
_animation->_a_over.start(0, st::stickersRowDuration);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1513,6 +1524,7 @@ QSize LayoutInlineGif::countFrameSize() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
LayoutInlineGif::~LayoutInlineGif() {
|
LayoutInlineGif::~LayoutInlineGif() {
|
||||||
|
if (gif()) deleteAndMark(_gif);
|
||||||
deleteAndMark(_animation);
|
deleteAndMark(_animation);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1591,12 +1603,6 @@ void LayoutInlineGif::clipCallback(ClipReaderNotification notification) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void LayoutInlineGif::update() {
|
|
||||||
if (_position >= 0) {
|
|
||||||
Ui::repaintInlineItem(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int32 LayoutInlineGif::content_width() const {
|
int32 LayoutInlineGif::content_width() const {
|
||||||
DocumentData *doc = _doc ? _doc : (_result ? _result->doc : 0);
|
DocumentData *doc = _doc ? _doc : (_result ? _result->doc : 0);
|
||||||
if (doc) {
|
if (doc) {
|
||||||
|
@ -1674,3 +1680,127 @@ QByteArray LayoutInlineGif::content_data() const {
|
||||||
DocumentData *doc = _doc ? _doc : (_result ? _result->doc : 0);
|
DocumentData *doc = _doc ? _doc : (_result ? _result->doc : 0);
|
||||||
return doc ? doc->data() : _result->data();
|
return doc ? doc->data() : _result->data();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LayoutInlinePhoto::LayoutInlinePhoto(InlineResult *result, PhotoData *photo) : LayoutInlineItem(result, 0, photo)
|
||||||
|
, _send(new SendInlineItemLink())
|
||||||
|
, _thumbLoaded(false) {
|
||||||
|
}
|
||||||
|
|
||||||
|
void LayoutInlinePhoto::initDimensions() {
|
||||||
|
int32 w = content_width(), h = content_height();
|
||||||
|
if (w <= 0 || h <= 0) {
|
||||||
|
_maxw = 0;
|
||||||
|
} else {
|
||||||
|
w = w * st::inlineMediaHeight / h;
|
||||||
|
_maxw = qMax(w, int32(st::inlineResultsMinWidth));
|
||||||
|
}
|
||||||
|
_minh = st::inlineMediaHeight + st::inlineResultsSkip;
|
||||||
|
}
|
||||||
|
|
||||||
|
void LayoutInlinePhoto::paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const {
|
||||||
|
bool loaded = content_loaded();
|
||||||
|
|
||||||
|
int32 height = st::inlineMediaHeight;
|
||||||
|
QSize frame = countFrameSize();
|
||||||
|
|
||||||
|
QRect r(0, 0, _width, height);
|
||||||
|
|
||||||
|
prepareThumb(_width, height, frame);
|
||||||
|
if (_thumb.isNull()) {
|
||||||
|
p.fillRect(r, st::overviewPhotoBg);
|
||||||
|
} else {
|
||||||
|
p.drawPixmap(r.topLeft(), _thumb);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void LayoutInlinePhoto::getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const {
|
||||||
|
if (x >= 0 && x < _width && y >= 0 && y < st::inlineMediaHeight) {
|
||||||
|
link = _send;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QSize LayoutInlinePhoto::countFrameSize() const {
|
||||||
|
int32 framew = content_width(), frameh = content_height(), height = st::inlineMediaHeight;
|
||||||
|
if (framew * height > frameh * _width) {
|
||||||
|
if (framew < st::maxStickerSize || frameh > height) {
|
||||||
|
if (frameh > height || (framew * height / frameh) <= st::maxStickerSize) {
|
||||||
|
framew = framew * height / frameh;
|
||||||
|
frameh = height;
|
||||||
|
} else {
|
||||||
|
frameh = int32(frameh * st::maxStickerSize) / framew;
|
||||||
|
framew = st::maxStickerSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (frameh < st::maxStickerSize || framew > _width) {
|
||||||
|
if (framew > _width || (frameh * _width / framew) <= st::maxStickerSize) {
|
||||||
|
frameh = frameh * _width / framew;
|
||||||
|
framew = _width;
|
||||||
|
} else {
|
||||||
|
framew = int32(framew * st::maxStickerSize) / frameh;
|
||||||
|
frameh = st::maxStickerSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return QSize(framew, frameh);
|
||||||
|
}
|
||||||
|
|
||||||
|
void LayoutInlinePhoto::prepareThumb(int32 width, int32 height, const QSize &frame) const {
|
||||||
|
if (_photo) {
|
||||||
|
if (_photo->medium->loaded()) {
|
||||||
|
if (!_thumbLoaded || _thumb.width() != width * cIntRetinaFactor() || _thumb.height() != height * cIntRetinaFactor()) {
|
||||||
|
_thumb = _photo->medium->pixNoCache(frame.width() * cIntRetinaFactor(), frame.height() * cIntRetinaFactor(), true, false, false, width, height);
|
||||||
|
}
|
||||||
|
_thumbLoaded = true;
|
||||||
|
} else {
|
||||||
|
if (_photo->thumb->loaded()) {
|
||||||
|
if (_thumb.width() != width * cIntRetinaFactor() || _thumb.height() != height * cIntRetinaFactor()) {
|
||||||
|
_thumb = _photo->thumb->pixNoCache(frame.width() * cIntRetinaFactor(), frame.height() * cIntRetinaFactor(), true, false, false, width, height);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_photo->medium->load();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (_result->thumb->loaded()) {
|
||||||
|
if (_thumb.width() != width * cIntRetinaFactor() || _thumb.height() != height * cIntRetinaFactor()) {
|
||||||
|
_thumb = _result->thumb->pixNoCache(frame.width() * cIntRetinaFactor(), frame.height() * cIntRetinaFactor(), true, false, false, width, height);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
_result->thumb->load();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int32 LayoutInlinePhoto::content_width() const {
|
||||||
|
PhotoData *photo = _photo ? _photo : (_result ? _result->photo : 0);
|
||||||
|
if (photo) {
|
||||||
|
return photo->full->width();
|
||||||
|
} else if (_result) {
|
||||||
|
return _result->width;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32 LayoutInlinePhoto::content_height() const {
|
||||||
|
PhotoData *photo = _photo ? _photo : (_result ? _result->photo : 0);
|
||||||
|
if (photo) {
|
||||||
|
return photo->full->height();
|
||||||
|
} else if (_result) {
|
||||||
|
return _result->height;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool LayoutInlinePhoto::content_loaded() const {
|
||||||
|
PhotoData *photo = _photo ? _photo : (_result ? _result->photo : 0);
|
||||||
|
return photo ? photo->loaded() : _result->loaded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void LayoutInlinePhoto::content_forget() {
|
||||||
|
PhotoData *photo = _photo ? _photo : (_result ? _result->photo : 0);
|
||||||
|
if (photo) {
|
||||||
|
photo->forget();
|
||||||
|
} else {
|
||||||
|
_result->forget();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -502,6 +502,8 @@ public:
|
||||||
PhotoData *photo() const;
|
PhotoData *photo() const;
|
||||||
void preload();
|
void preload();
|
||||||
|
|
||||||
|
void update();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
InlineResult *_result;
|
InlineResult *_result;
|
||||||
DocumentData *_doc;
|
DocumentData *_doc;
|
||||||
|
@ -576,7 +578,7 @@ private:
|
||||||
bool gif() const {
|
bool gif() const {
|
||||||
return (!_gif || _gif == BadClipReader) ? false : true;
|
return (!_gif || _gif == BadClipReader) ? false : true;
|
||||||
}
|
}
|
||||||
QPixmap _thumb;
|
mutable QPixmap _thumb;
|
||||||
void prepareThumb(int32 width, int32 height, const QSize &frame) const;
|
void prepareThumb(int32 width, int32 height, const QSize &frame) const;
|
||||||
|
|
||||||
void ensureAnimation() const;
|
void ensureAnimation() const;
|
||||||
|
@ -584,7 +586,6 @@ private:
|
||||||
void step_radial(uint64 ms, bool timer);
|
void step_radial(uint64 ms, bool timer);
|
||||||
|
|
||||||
void clipCallback(ClipReaderNotification notification);
|
void clipCallback(ClipReaderNotification notification);
|
||||||
void update();
|
|
||||||
|
|
||||||
struct AnimationData {
|
struct AnimationData {
|
||||||
AnimationData(AnimationCreator creator)
|
AnimationData(AnimationCreator creator)
|
||||||
|
@ -599,3 +600,32 @@ private:
|
||||||
mutable FloatAnimation _a_deleteOver;
|
mutable FloatAnimation _a_deleteOver;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class LayoutInlinePhoto : public LayoutInlineItem {
|
||||||
|
public:
|
||||||
|
LayoutInlinePhoto(InlineResult *result, PhotoData *photo);
|
||||||
|
|
||||||
|
virtual void initDimensions();
|
||||||
|
|
||||||
|
virtual bool fullLine() const {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const;
|
||||||
|
virtual void getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
QSize countFrameSize() const;
|
||||||
|
|
||||||
|
int32 content_width() const;
|
||||||
|
int32 content_height() const;
|
||||||
|
bool content_loaded() const;
|
||||||
|
void content_forget();
|
||||||
|
|
||||||
|
TextLinkPtr _send;
|
||||||
|
|
||||||
|
mutable QPixmap _thumb;
|
||||||
|
mutable bool _thumbLoaded;
|
||||||
|
void prepareThumb(int32 width, int32 height, const QSize &frame) const;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
|
@ -2576,6 +2576,15 @@ namespace Local {
|
||||||
return _stickerImagesMap.constFind(location) != _stickerImagesMap.cend();
|
return _stickerImagesMap.constFind(location) != _stickerImagesMap.cend();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void copyStickerImage(const StorageKey &oldLocation, const StorageKey &newLocation) {
|
||||||
|
StorageMap::const_iterator i = _stickerImagesMap.constFind(oldLocation);
|
||||||
|
if (i != _stickerImagesMap.cend()) {
|
||||||
|
_stickerImagesMap.insert(newLocation, i.value());
|
||||||
|
_mapChanged = true;
|
||||||
|
_writeMap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int32 hasStickers() {
|
int32 hasStickers() {
|
||||||
return _stickerImagesMap.size();
|
return _stickerImagesMap.size();
|
||||||
}
|
}
|
||||||
|
|
|
@ -130,6 +130,7 @@ namespace Local {
|
||||||
void writeStickerImage(const StorageKey &location, const QByteArray &data, bool overwrite = true);
|
void writeStickerImage(const StorageKey &location, const QByteArray &data, bool overwrite = true);
|
||||||
TaskId startStickerImageLoad(const StorageKey &location, mtpFileLoader *loader);
|
TaskId startStickerImageLoad(const StorageKey &location, mtpFileLoader *loader);
|
||||||
bool willStickerImageLoad(const StorageKey &location);
|
bool willStickerImageLoad(const StorageKey &location);
|
||||||
|
void copyStickerImage(const StorageKey &oldLocation, const StorageKey &newLocation);
|
||||||
int32 hasStickers();
|
int32 hasStickers();
|
||||||
qint64 storageStickersSize();
|
qint64 storageStickersSize();
|
||||||
|
|
||||||
|
|
|
@ -1888,6 +1888,24 @@ void MainWidget::documentLoadRetry() {
|
||||||
if (document) document->save(failedFileName);
|
if (document) document->save(failedFileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MainWidget::inlineResultLoadProgress(FileLoader *loader) {
|
||||||
|
//InlineResult *result = App::inlineResultFromLoader(loader);
|
||||||
|
//if (!result) return;
|
||||||
|
|
||||||
|
//result->loaded();
|
||||||
|
|
||||||
|
//Ui::repaintInlineItem();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MainWidget::inlineResultLoadFailed(FileLoader *loader, bool started) {
|
||||||
|
//InlineResult *result = App::inlineResultFromLoader(loader);
|
||||||
|
//if (!result) return;
|
||||||
|
|
||||||
|
//result->loaded();
|
||||||
|
|
||||||
|
//Ui::repaintInlineItem();
|
||||||
|
}
|
||||||
|
|
||||||
void MainWidget::audioMarkRead(AudioData *data) {
|
void MainWidget::audioMarkRead(AudioData *data) {
|
||||||
const AudioItems &items(App::audioItems());
|
const AudioItems &items(App::audioItems());
|
||||||
AudioItems::const_iterator i = items.constFind(data);
|
AudioItems::const_iterator i = items.constFind(data);
|
||||||
|
|
|
@ -450,6 +450,8 @@ public slots:
|
||||||
void documentLoadFailed(FileLoader *loader, bool started);
|
void documentLoadFailed(FileLoader *loader, bool started);
|
||||||
void documentLoadRetry();
|
void documentLoadRetry();
|
||||||
void documentPlayProgress(const SongMsgId &songId);
|
void documentPlayProgress(const SongMsgId &songId);
|
||||||
|
void inlineResultLoadProgress(FileLoader *loader);
|
||||||
|
void inlineResultLoadFailed(FileLoader *loader, bool started);
|
||||||
void hidePlayer();
|
void hidePlayer();
|
||||||
|
|
||||||
void dialogsCancelled();
|
void dialogsCancelled();
|
||||||
|
|
|
@ -57,7 +57,7 @@ namespace {
|
||||||
WebLoadMainManager *_webLoadMainManager = 0;
|
WebLoadMainManager *_webLoadMainManager = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
FileLoader::FileLoader(const QString &toFile, int32 size, LocationType locationType, LoadFromCloudSetting fromCloud, bool autoLoading)
|
FileLoader::FileLoader(const QString &toFile, int32 size, LocationType locationType, LoadToCacheSetting toCache, LoadFromCloudSetting fromCloud, bool autoLoading)
|
||||||
: _prev(0)
|
: _prev(0)
|
||||||
, _next(0)
|
, _next(0)
|
||||||
, _priority(0)
|
, _priority(0)
|
||||||
|
@ -67,7 +67,7 @@ FileLoader::FileLoader(const QString &toFile, int32 size, LocationType locationT
|
||||||
, _complete(false)
|
, _complete(false)
|
||||||
, _localStatus(LocalNotTried)
|
, _localStatus(LocalNotTried)
|
||||||
, _fileIsOpen(false)
|
, _fileIsOpen(false)
|
||||||
, _toCache(LoadToCacheAsWell)
|
, _toCache(toCache)
|
||||||
, _fromCloud(fromCloud)
|
, _fromCloud(fromCloud)
|
||||||
, _size(size)
|
, _size(size)
|
||||||
, _type(mtpc_storage_fileUnknown)
|
, _type(mtpc_storage_fileUnknown)
|
||||||
|
@ -339,7 +339,7 @@ void FileLoader::startLoading(bool loadFirst, bool prior) {
|
||||||
}
|
}
|
||||||
|
|
||||||
mtpFileLoader::mtpFileLoader(const StorageImageLocation *location, int32 size, LoadFromCloudSetting fromCloud, bool autoLoading)
|
mtpFileLoader::mtpFileLoader(const StorageImageLocation *location, int32 size, LoadFromCloudSetting fromCloud, bool autoLoading)
|
||||||
: FileLoader(QString(), size, UnknownFileLocation, fromCloud, autoLoading)
|
: FileLoader(QString(), size, UnknownFileLocation, LoadToCacheAsWell, fromCloud, autoLoading)
|
||||||
, _lastComplete(false)
|
, _lastComplete(false)
|
||||||
, _skippedBytes(0)
|
, _skippedBytes(0)
|
||||||
, _nextRequestOffset(0)
|
, _nextRequestOffset(0)
|
||||||
|
@ -355,7 +355,7 @@ mtpFileLoader::mtpFileLoader(const StorageImageLocation *location, int32 size, L
|
||||||
}
|
}
|
||||||
|
|
||||||
mtpFileLoader::mtpFileLoader(int32 dc, const uint64 &id, const uint64 &access, LocationType type, const QString &to, int32 size, LoadToCacheSetting toCache, LoadFromCloudSetting fromCloud, bool autoLoading)
|
mtpFileLoader::mtpFileLoader(int32 dc, const uint64 &id, const uint64 &access, LocationType type, const QString &to, int32 size, LoadToCacheSetting toCache, LoadFromCloudSetting fromCloud, bool autoLoading)
|
||||||
: FileLoader(to, size, type, fromCloud, autoLoading)
|
: FileLoader(to, size, type, toCache, fromCloud, autoLoading)
|
||||||
, _lastComplete(false)
|
, _lastComplete(false)
|
||||||
, _skippedBytes(0)
|
, _skippedBytes(0)
|
||||||
, _nextRequestOffset(0)
|
, _nextRequestOffset(0)
|
||||||
|
@ -566,7 +566,7 @@ mtpFileLoader::~mtpFileLoader() {
|
||||||
}
|
}
|
||||||
|
|
||||||
webFileLoader::webFileLoader(const QString &url, const QString &to, LoadFromCloudSetting fromCloud, bool autoLoading)
|
webFileLoader::webFileLoader(const QString &url, const QString &to, LoadFromCloudSetting fromCloud, bool autoLoading)
|
||||||
: FileLoader(QString(), 0, UnknownFileLocation, fromCloud, autoLoading)
|
: FileLoader(QString(), 0, UnknownFileLocation, LoadToCacheAsWell, fromCloud, autoLoading)
|
||||||
, _url(url)
|
, _url(url)
|
||||||
, _requestSent(false)
|
, _requestSent(false)
|
||||||
, _already(0) {
|
, _already(0) {
|
||||||
|
@ -835,6 +835,8 @@ void WebLoadManager::onFailed(QNetworkReply *reply) {
|
||||||
webFileLoaderPrivate *loader = j.value();
|
webFileLoaderPrivate *loader = j.value();
|
||||||
_replies.erase(j);
|
_replies.erase(j);
|
||||||
|
|
||||||
|
LOG(("Network Error: Failed to request '%1', error %2 (%3)").arg(QString::fromLatin1(loader->_url.toEncoded())).arg(int(reply->error())).arg(reply->errorString()));
|
||||||
|
|
||||||
if (!handleReplyResult(loader, WebReplyProcessError)) {
|
if (!handleReplyResult(loader, WebReplyProcessError)) {
|
||||||
_loaders.remove(loader);
|
_loaders.remove(loader);
|
||||||
delete loader;
|
delete loader;
|
||||||
|
|
|
@ -126,7 +126,7 @@ class FileLoader : public QObject {
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
FileLoader(const QString &toFile, int32 size, LocationType locationType, LoadFromCloudSetting fromCloud, bool autoLoading);
|
FileLoader(const QString &toFile, int32 size, LocationType locationType, LoadToCacheSetting, LoadFromCloudSetting fromCloud, bool autoLoading);
|
||||||
bool done() const {
|
bool done() const {
|
||||||
return _complete;
|
return _complete;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1514,8 +1514,10 @@ void DocumentData::setattributes(const QVector<MTPDocumentAttribute> &attributes
|
||||||
type = StickerDocument;
|
type = StickerDocument;
|
||||||
StickerData *sticker = new StickerData();
|
StickerData *sticker = new StickerData();
|
||||||
_additional = sticker;
|
_additional = sticker;
|
||||||
sticker->alt = qs(d.valt);
|
}
|
||||||
sticker->set = d.vstickerset;
|
if (sticker()) {
|
||||||
|
sticker()->alt = qs(d.valt);
|
||||||
|
sticker()->set = d.vstickerset;
|
||||||
}
|
}
|
||||||
} break;
|
} break;
|
||||||
case mtpc_documentAttributeVideo: {
|
case mtpc_documentAttributeVideo: {
|
||||||
|
@ -1528,12 +1530,16 @@ void DocumentData::setattributes(const QVector<MTPDocumentAttribute> &attributes
|
||||||
} break;
|
} break;
|
||||||
case mtpc_documentAttributeAudio: {
|
case mtpc_documentAttributeAudio: {
|
||||||
const MTPDdocumentAttributeAudio &d(attributes[i].c_documentAttributeAudio());
|
const MTPDdocumentAttributeAudio &d(attributes[i].c_documentAttributeAudio());
|
||||||
type = SongDocument;
|
if (type == FileDocument) {
|
||||||
SongData *song = new SongData();
|
type = SongDocument;
|
||||||
_additional = song;
|
SongData *song = new SongData();
|
||||||
song->duration = d.vduration.v;
|
_additional = song;
|
||||||
song->title = qs(d.vtitle);
|
}
|
||||||
song->performer = qs(d.vperformer);
|
if (song()) {
|
||||||
|
song()->duration = d.vduration.v;
|
||||||
|
song()->title = qs(d.vtitle);
|
||||||
|
song()->performer = qs(d.vperformer);
|
||||||
|
}
|
||||||
} break;
|
} break;
|
||||||
case mtpc_documentAttributeFilename: name = qs(attributes[i].c_documentAttributeFilename().vfile_name); break;
|
case mtpc_documentAttributeFilename: name = qs(attributes[i].c_documentAttributeFilename().vfile_name); break;
|
||||||
}
|
}
|
||||||
|
@ -2034,8 +2040,53 @@ void ImageLinkData::load() {
|
||||||
manager.getData(this);
|
manager.getData(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InlineResult::automaticLoadGif() const {
|
void InlineResult::automaticLoadGif() {
|
||||||
|
if (loaded() || type != qstr("gif") || (content_type != qstr("video/mp4") && content_type != "image/gif")) return;
|
||||||
|
|
||||||
|
if (_loader != CancelledWebFileLoader) {
|
||||||
|
// if load at least anywhere
|
||||||
|
bool loadFromCloud = !(cAutoDownloadGif() & dbiadNoPrivate) || !(cAutoDownloadGif() & dbiadNoGroups);
|
||||||
|
saveFile(QString(), loadFromCloud ? LoadFromCloudOrLocal : LoadFromLocalOnly, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void InlineResult::saveFile(const QString &toFile, LoadFromCloudSetting fromCloud, bool autoLoading) {
|
||||||
|
if (loaded()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_loader == CancelledWebFileLoader) _loader = 0;
|
||||||
|
if (_loader) {
|
||||||
|
if (!_loader->setFileName(toFile)) {
|
||||||
|
cancelFile();
|
||||||
|
_loader = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_loader) {
|
||||||
|
if (fromCloud == LoadFromCloudOrLocal) _loader->permitLoadFromCloud();
|
||||||
|
} else {
|
||||||
|
_loader = new webFileLoader(content_url, toFile, fromCloud, autoLoading);
|
||||||
|
App::regInlineResultLoader(_loader, this);
|
||||||
|
|
||||||
|
_loader->connect(_loader, SIGNAL(progress(FileLoader*)), App::main(), SLOT(inlineResultLoadProgress(FileLoader*)));
|
||||||
|
_loader->connect(_loader, SIGNAL(failed(FileLoader*,bool)), App::main(), SLOT(inlineResultLoadFailed(FileLoader*,bool)));
|
||||||
|
_loader->start();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void InlineResult::cancelFile() {
|
||||||
|
if (!loading()) return;
|
||||||
|
|
||||||
|
App::unregInlineResultLoader(_loader);
|
||||||
|
|
||||||
|
webFileLoader *l = _loader;
|
||||||
|
_loader = CancelledWebFileLoader;
|
||||||
|
if (l) {
|
||||||
|
l->cancel();
|
||||||
|
l->deleteLater();
|
||||||
|
l->stop();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QByteArray InlineResult::data() const {
|
QByteArray InlineResult::data() const {
|
||||||
|
@ -2043,22 +2094,43 @@ QByteArray InlineResult::data() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InlineResult::loading() const {
|
bool InlineResult::loading() const {
|
||||||
return false;
|
return _loader && _loader != CancelledWebFileLoader;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InlineResult::loaded() const {
|
bool InlineResult::loaded() const {
|
||||||
return false;
|
if (loading() && _loader->done()) {
|
||||||
|
App::unregInlineResultLoader(_loader);
|
||||||
|
if (_loader->fileType() == mtpc_storage_fileUnknown) {
|
||||||
|
_loader->deleteLater();
|
||||||
|
_loader->stop();
|
||||||
|
_loader = CancelledWebFileLoader;
|
||||||
|
} else {
|
||||||
|
InlineResult *that = const_cast<InlineResult*>(this);
|
||||||
|
that->_data = _loader->bytes();
|
||||||
|
|
||||||
|
_loader->deleteLater();
|
||||||
|
_loader->stop();
|
||||||
|
_loader = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return !_data.isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InlineResult::displayLoading() const {
|
bool InlineResult::displayLoading() const {
|
||||||
return false;
|
return loading() ? (!_loader->loadingLocal() || !_loader->autoLoading()) : false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void InlineResult::forget() {
|
void InlineResult::forget() {
|
||||||
|
thumb->forget();
|
||||||
|
_data.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
float64 InlineResult::progress() const {
|
float64 InlineResult::progress() const {
|
||||||
return 0.;
|
return loading() ? _loader->currentProgress() : (loaded() ? 1 : 0); return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
InlineResult::~InlineResult() {
|
||||||
|
cancelFile();
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerLink::onClick(Qt::MouseButton button) const {
|
void PeerLink::onClick(Qt::MouseButton button) const {
|
||||||
|
|
|
@ -1346,7 +1346,8 @@ public:
|
||||||
, width(0)
|
, width(0)
|
||||||
, height(0)
|
, height(0)
|
||||||
, duration(0)
|
, duration(0)
|
||||||
, noWebPage(false) {
|
, noWebPage(false)
|
||||||
|
, _loader(0) {
|
||||||
}
|
}
|
||||||
uint64 queryId;
|
uint64 queryId;
|
||||||
QString id, type;
|
QString id, type;
|
||||||
|
@ -1362,7 +1363,11 @@ public:
|
||||||
QString caption; // if message.isEmpty() use botContextMessageMediaAuto
|
QString caption; // if message.isEmpty() use botContextMessageMediaAuto
|
||||||
|
|
||||||
ImagePtr thumb;
|
ImagePtr thumb;
|
||||||
void automaticLoadGif() const;
|
|
||||||
|
void automaticLoadGif();
|
||||||
|
void saveFile(const QString &toFile, LoadFromCloudSetting fromCloud, bool autoLoading);
|
||||||
|
void cancelFile();
|
||||||
|
|
||||||
QByteArray data() const;
|
QByteArray data() const;
|
||||||
bool loading() const;
|
bool loading() const;
|
||||||
bool loaded() const;
|
bool loaded() const;
|
||||||
|
@ -1370,8 +1375,11 @@ public:
|
||||||
void forget();
|
void forget();
|
||||||
float64 progress() const;
|
float64 progress() const;
|
||||||
|
|
||||||
|
~InlineResult();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QByteArray _data;
|
QByteArray _data;
|
||||||
|
mutable webFileLoader *_loader;
|
||||||
|
|
||||||
};
|
};
|
||||||
typedef QList<InlineResult*> InlineResults;
|
typedef QList<InlineResult*> InlineResults;
|
||||||
|
|
|
@ -556,19 +556,19 @@ template <typename R>
|
||||||
inline FunctionCreator<R> func(R(*method)()) {
|
inline FunctionCreator<R> func(R(*method)()) {
|
||||||
return FunctionCreator<R>(new WrappedFunction<R>(method));
|
return FunctionCreator<R>(new WrappedFunction<R>(method));
|
||||||
}
|
}
|
||||||
template <typename O, typename R>
|
template <typename O, typename I, typename R>
|
||||||
class ObjectFunction : public FunctionImplementation<R> {
|
class ObjectFunction : public FunctionImplementation<R> {
|
||||||
public:
|
public:
|
||||||
typedef R(O::*Method)();
|
typedef R(I::*Method)();
|
||||||
ObjectFunction(O *obj, Method method) : _obj(obj), _method(method) {}
|
ObjectFunction(O *obj, Method method) : _obj(obj), _method(method) {}
|
||||||
virtual R call() { return (_obj->*_method)(); }
|
virtual R call() { return (_obj->*_method)(); }
|
||||||
private:
|
private:
|
||||||
O *_obj;
|
O *_obj;
|
||||||
Method _method;
|
Method _method;
|
||||||
};
|
};
|
||||||
template <typename O, typename R>
|
template <typename O, typename I, typename R>
|
||||||
inline FunctionCreator<R> func(O *obj, R(O::*method)()) {
|
inline FunctionCreator<R> func(O *obj, R(I::*method)()) {
|
||||||
return FunctionCreator<R>(new ObjectFunction<O, R>(obj, method));
|
return FunctionCreator<R>(new ObjectFunction<O, I, R>(obj, method));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename R, typename A1>
|
template <typename R, typename A1>
|
||||||
|
@ -625,19 +625,19 @@ template <typename R, typename A1>
|
||||||
inline Function1Creator<R, A1> func(R(*method)(A1)) {
|
inline Function1Creator<R, A1> func(R(*method)(A1)) {
|
||||||
return Function1Creator<R, A1>(new WrappedFunction1<R, A1>(method));
|
return Function1Creator<R, A1>(new WrappedFunction1<R, A1>(method));
|
||||||
}
|
}
|
||||||
template <typename O, typename R, typename A1>
|
template <typename O, typename I, typename R, typename A1>
|
||||||
class ObjectFunction1 : public Function1Implementation<R, A1> {
|
class ObjectFunction1 : public Function1Implementation<R, A1> {
|
||||||
public:
|
public:
|
||||||
typedef R(O::*Method)(A1);
|
typedef R(I::*Method)(A1);
|
||||||
ObjectFunction1(O *obj, Method method) : _obj(obj), _method(method) {}
|
ObjectFunction1(O *obj, Method method) : _obj(obj), _method(method) {}
|
||||||
virtual R call(A1 a1) { return (_obj->*_method)(a1); }
|
virtual R call(A1 a1) { return (_obj->*_method)(a1); }
|
||||||
private:
|
private:
|
||||||
O *_obj;
|
O *_obj;
|
||||||
Method _method;
|
Method _method;
|
||||||
};
|
};
|
||||||
template <typename O, typename R, typename A1>
|
template <typename O, typename I, typename R, typename A1>
|
||||||
Function1Creator<R, A1> func(O *obj, R(O::*method)(A1)) {
|
Function1Creator<R, A1> func(O *obj, R(I::*method)(A1)) {
|
||||||
return Function1Creator<R, A1>(new ObjectFunction1<O, R, A1>(obj, method));
|
return Function1Creator<R, A1>(new ObjectFunction1<O, I, R, A1>(obj, method));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename R, typename A1, typename A2>
|
template <typename R, typename A1, typename A2>
|
||||||
|
@ -695,17 +695,17 @@ Function2Creator<R, A1, A2> func(R(*method)(A1, A2)) {
|
||||||
return Function2Creator<R, A1, A2>(new WrappedFunction2<R, A1, A2>(method));
|
return Function2Creator<R, A1, A2>(new WrappedFunction2<R, A1, A2>(method));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename O, typename R, typename A1, typename A2>
|
template <typename O, typename I, typename R, typename A1, typename A2>
|
||||||
class ObjectFunction2 : public Function2Implementation<R, A1, A2> {
|
class ObjectFunction2 : public Function2Implementation<R, A1, A2> {
|
||||||
public:
|
public:
|
||||||
typedef R(O::*Method)(A1, A2);
|
typedef R(I::*Method)(A1, A2);
|
||||||
ObjectFunction2(O *obj, Method method) : _obj(obj), _method(method) {}
|
ObjectFunction2(O *obj, Method method) : _obj(obj), _method(method) {}
|
||||||
virtual R call(A1 a1, A2 a2) { return (_obj->*_method)(a1, a2); }
|
virtual R call(A1 a1, A2 a2) { return (_obj->*_method)(a1, a2); }
|
||||||
private:
|
private:
|
||||||
O *_obj;
|
O *_obj;
|
||||||
Method _method;
|
Method _method;
|
||||||
};
|
};
|
||||||
template <typename O, typename R, typename A1, typename A2>
|
template <typename O, typename I, typename R, typename A1, typename A2>
|
||||||
Function2Creator<R, A1, A2> func(O *obj, R(O::*method)(A1, A2)) {
|
Function2Creator<R, A1, A2> func(O *obj, R(I::*method)(A1, A2)) {
|
||||||
return Function2Creator<R, A1, A2>(new ObjectFunction2<O, R, A1, A2>(obj, method));
|
return Function2Creator<R, A1, A2>(new ObjectFunction2<O, I, R, A1, A2>(obj, method));
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue