/*
Copyright (C) 2012-24 Miranda NG team (https://miranda-ng.org)
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation version 2
of the License.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
#include "stdafx.h"
enum class BBCODE
{
BOLD, ITALIC, STRIKE, UNDERLINE, URL, CODE, QUOTE
};
struct
{
BBCODE type;
const wchar_t *begin, *end;
unsigned len1, len2;
}
static bbCodes[] =
{
{ BBCODE::BOLD, L"[b]", L"[/b]", 3, 4 },
{ BBCODE::ITALIC, L"[i]", L"[/i]", 3, 4 },
{ BBCODE::STRIKE, L"[s]", L"[/s]", 3, 4 },
{ BBCODE::UNDERLINE, L"[u]", L"[/u]", 3, 4 },
{ BBCODE::URL, L"[url]", L"[/url]", 5, 6 },
{ BBCODE::CODE, L"[code]", L"[/code]", 6, 7 },
{ BBCODE::QUOTE, L"[quote]", L"[/quote]", 7, 8 },
};
TD::object_ptr formatBbcodes(const char *pszText)
{
auto res = TD::make_object();
if (mir_strlen(pszText)) {
std::wstring str = Utf2T(pszText).get();
for (auto &it : bbCodes) {
while (true) {
int i1 = (int)str.find(it.begin);
if (i1 == str.npos)
break;
int i2 = (int)str.find(it.end, i1);
if (i2 == str.npos)
break;
for (auto &jt : res->entities_) {
if (jt->offset_ > i1)
jt->offset_ -= it.len1;
if (jt->offset_ > i2)
jt->offset_ -= it.len2;
}
str.erase(i2, it.len2); i2 -= it.len1;
str.erase(i1, it.len1);
TD::object_ptr pNew;
switch (it.type) {
case BBCODE::URL: pNew = TD::make_object(); break;
case BBCODE::CODE: pNew = TD::make_object(); break;
case BBCODE::BOLD: pNew = TD::make_object(); break;
case BBCODE::QUOTE: pNew = TD::make_object(); break;
case BBCODE::ITALIC: pNew = TD::make_object(); break;
case BBCODE::STRIKE: pNew = TD::make_object(); break;
case BBCODE::UNDERLINE: pNew = TD::make_object(); break;
}
res->entities_.push_back(TD::make_object(TD::int32(i1), TD::int32(i2 - i1), std::move(pNew)));
}
}
res->text_ = T2Utf(str.c_str()).get();
}
return res;
}
CMStringA CTelegramProto::GetFormattedText(TD::object_ptr &pText)
{
CMStringW ret(Utf2T(pText->text_.c_str()));
unsigned offset = 0;
for (auto &it : pText->entities_) {
int iCode;
switch (it->type_->get_id()) {
case TD::textEntityTypeBold::ID: iCode = 0; break;
case TD::textEntityTypeItalic::ID: iCode = 1; break;
case TD::textEntityTypeStrikethrough::ID: iCode = 2; break;
case TD::textEntityTypeUnderline::ID: iCode = 3; break;
case TD::textEntityTypeTextUrl::ID: iCode = 4; break;
case TD::textEntityTypeCode::ID: iCode = 5; break;
case TD::textEntityTypeBlockQuote::ID: iCode = 6; break;
default:
continue;
}
auto &bb = bbCodes[iCode];
ret.Insert(offset + it->offset_ + it->length_, bb.end);
ret.Insert(offset + it->offset_, bb.begin);
offset += bb.len1 + bb.len2;
}
return T2Utf(ret).get();
}
/////////////////////////////////////////////////////////////////////////////////////////
CMStringA msg2id(TD::int53 chatId, TD::int53 msgId)
{
return CMStringA(FORMAT, "%lld_%lld", chatId, msgId);
}
CMStringA msg2id(const TD::message *pMsg)
{
return CMStringA(FORMAT, "%lld_%lld", pMsg->chat_id_, pMsg->id_);
}
TD::int53 dbei2id(const DBEVENTINFO &dbei)
{
if (dbei.szId == nullptr)
return -1;
auto *p = strchr(dbei.szId, '_');
return _atoi64(p ? p + 1 : dbei.szId);
}
/////////////////////////////////////////////////////////////////////////////////////////
const char *getName(const TD::usernames *pName)
{
return (pName == nullptr) ? TranslateU("none") : pName->editable_username_.c_str();
}
TD::object_ptr makeFile(const wchar_t *pwszFilename)
{
std::string szPath = T2Utf(pwszFilename);
return TD::make_object(std::move(szPath));
}
TG_FILE_REQUEST::Type AutoDetectType(const wchar_t *pwszFilename)
{
if (ProtoGetAvatarFileFormat(pwszFilename) != PA_FORMAT_UNKNOWN)
return TG_FILE_REQUEST::PICTURE;
CMStringW path(pwszFilename);
int idx = path.ReverseFind('.');
if (idx == -1 || path.Find('\\', idx) != -1)
return TG_FILE_REQUEST::FILE;
auto wszExt = path.Right(path.GetLength() - idx);
wszExt.MakeLower();
if (wszExt == L"mp4" || wszExt == L"webm")
return TG_FILE_REQUEST::VIDEO;
else if (wszExt == L"mp3" || wszExt == "ogg" || wszExt == "oga" || wszExt == "wav")
return TG_FILE_REQUEST::VOICE;
return TG_FILE_REQUEST::FILE;
}
TD::int53 getReplyId(const TD::MessageReplyTo *pReply)
{
if (pReply) {
switch (pReply->get_id()) {
case TD::messageReplyToMessage::ID:
return ((TD::messageReplyToMessage *)pReply)->message_id_;
case TD::inputMessageReplyToExternalMessage::ID:
return ((TD::inputMessageReplyToExternalMessage *)pReply)->message_id_;
}
}
return 0;
}
CMStringW TG_USER::getDisplayName() const
{
if (hContact != 0) {
if (hContact != INVALID_CONTACT_ID)
return Clist_GetContactDisplayName(hContact, 0);
if (!wszFirstName.IsEmpty())
return (wszLastName.IsEmpty()) ? wszFirstName : wszFirstName + L" " + wszLastName;
}
return wszNick;
}
void CTelegramProto::RemoveFromClist(TG_USER *pUser)
{
Contact::RemoveFromList(pUser->hContact);
if (pUser->isForum) {
if (MGROUP hGroup = Clist_GroupExists(ptrW(Clist_GetGroup(pUser->hContact))))
Clist_GroupDelete(hGroup, true);
for (auto &cc : AccContacts())
if (pUser->id == GetId(cc, DBKEY_OWNER))
Contact::RemoveFromList(cc);
}
}
void CTelegramProto::MarkRead(MCONTACT hContact, const CMStringA &szMaxId, bool bSent)
{
for (MEVENT hEvent = db_event_firstUnread(hContact); hEvent; hEvent = db_event_next(hContact, hEvent)) {
DB::EventInfo dbei(hEvent, false);
if (!dbei || !dbei.szId)
continue;
if (dbei.szId > szMaxId)
break;
bool isSent = (dbei.flags & DBEF_SENT) != 0;
if (isSent != bSent)
continue;
if (!dbei.markedRead())
db_event_markRead(hContact, hEvent, true);
}
}
int CTelegramProto::GetDefaultMute(const TG_USER *pUser)
{
if (pUser->isGroupChat)
return (pUser->isChannel) ? m_iDefaultMuteChannel : m_iDefaultMuteGroup;
return m_iDefaultMutePrivate;
}
MCONTACT CTelegramProto::GetRealContact(const TG_USER *pUser)
{
return (pUser->hContact != 0) ? pUser->hContact : m_iSavedMessages;
}
TG_USER* CTelegramProto::GetSender(const TD::MessageSender *pSender)
{
switch (pSender->get_id()) {
case TD::messageSenderChat::ID:
return FindChat(((TD::messageSenderChat *)pSender)->chat_id_);
case TD::messageSenderUser::ID:
return FindUser(((TD::messageSenderUser *)pSender)->user_id_);
}
return nullptr;
}
/////////////////////////////////////////////////////////////////////////////////////////
bool CTelegramProto::CheckSearchUser(TG_USER *pUser)
{
auto pSearchId = std::find(m_searchIds.begin(), m_searchIds.end(), pUser->chatId);
if (pSearchId == m_searchIds.end())
return false;
ReportSearchUser(pUser);
m_searchIds.erase(pSearchId);
if (m_searchIds.empty())
ProtoBroadcastAck(0, ACKTYPE_SEARCH, ACKRESULT_SUCCESS, this);
return true;
}
void CTelegramProto::ReportSearchUser(TG_USER *pUser)
{
CMStringW wszId(FORMAT, L"%lld", pUser->id), wszNick, wszLastName, wszFirstName;
PROTOSEARCHRESULT psr = {};
psr.cbSize = sizeof(psr);
psr.flags = PSR_UNICODE;
psr.id.w = wszId.GetBuffer();
if (pUser->hContact != INVALID_CONTACT_ID) {
wszNick = getMStringW(pUser->hContact, "Nick");
wszLastName = getMStringW(pUser->hContact, "LastName");
wszFirstName = getMStringW(pUser->hContact, "FirstName");
psr.nick.w = wszNick.GetBuffer();
psr.lastName.w = wszLastName.GetBuffer();
psr.firstName.w = wszFirstName.GetBuffer();
}
else {
psr.firstName.w = pUser->wszFirstName.GetBuffer();
psr.lastName.w = pUser->wszLastName.GetBuffer();
psr.nick.w = pUser->wszNick.GetBuffer();
}
ProtoBroadcastAck(0, ACKTYPE_SEARCH, ACKRESULT_DATA, this, (LPARAM)&psr);
}
/////////////////////////////////////////////////////////////////////////////////////////
int64_t CTelegramProto::GetId(MCONTACT hContact, const char *pszSetting)
{
return _atoi64(getMStringA(hContact, pszSetting));
}
void CTelegramProto::SetId(MCONTACT hContact, int64_t id, const char *pszSetting)
{
char szId[100];
_i64toa(id, szId, 10);
setString(hContact, pszSetting, szId);
}
/////////////////////////////////////////////////////////////////////////////////////////
void CTelegramProto::UpdateString(MCONTACT hContact, const char *pszSetting, const std::string &str)
{
if (str.empty())
delSetting(hContact, pszSetting);
else
setUString(hContact, pszSetting, str.c_str());
}
/////////////////////////////////////////////////////////////////////////////////////////
// Users
TG_USER* CTelegramProto::FindChat(int64_t id)
{
auto *tmp = (TG_USER *)_alloca(sizeof(TG_USER));
tmp->chatId = id;
return m_arChats.find(tmp);
}
TG_USER* CTelegramProto::FindUser(int64_t id)
{
return m_arUsers.find((TG_USER *)&id);
}
TG_USER* CTelegramProto::AddFakeUser(int64_t id, bool bIsChat)
{
auto *pu = FindUser(id);
if (pu == nullptr) {
pu = new TG_USER(id, INVALID_CONTACT_ID, bIsChat);
m_arUsers.insert(pu);
}
return pu;
}
TG_USER* CTelegramProto::AddUser(int64_t id, bool bIsChat)
{
auto *pUser = FindUser(id);
if (pUser != nullptr)
if (pUser->hContact != INVALID_CONTACT_ID)
return pUser;
MCONTACT hContact = db_add_contact();
Proto_AddToContact(hContact, m_szModuleName);
SetId(hContact, id);
if (bIsChat) {
setByte(hContact, "ChatRoom", 1);
}
else if (mir_wstrlen(m_wszDefaultGroup))
Clist_SetGroup(hContact, m_wszDefaultGroup);
if (pUser == nullptr) {
pUser = new TG_USER(id, hContact, bIsChat);
m_arUsers.insert(pUser);
}
else {
pUser->hContact = hContact;
setWString(hContact, "Nick", pUser->wszNick);
if (!pUser->isGroupChat) {
setWString(hContact, "FirstName", pUser->wszFirstName);
setWString(hContact, "LastName", pUser->wszLastName);
}
else pUser->bStartChat = true;
}
return pUser;
}
/////////////////////////////////////////////////////////////////////////////////////////
// Popups
void CTelegramProto::InitPopups(void)
{
g_plugin.addPopupOption(CMStringW(FORMAT, TranslateT("%s error notifications"), m_tszUserName), m_bUsePopups);
char name[256];
mir_snprintf(name, "%s_%s", m_szModuleName, "Error");
wchar_t desc[256];
mir_snwprintf(desc, L"%s/%s", m_tszUserName, TranslateT("Errors"));
POPUPCLASS ppc = {};
ppc.flags = PCF_UNICODE;
ppc.pszName = name;
ppc.pszDescription.w = desc;
ppc.hIcon = IcoLib_GetIconByHandle(m_hProtoIcon);
ppc.colorBack = RGB(191, 0, 0); //Red
ppc.colorText = RGB(255, 245, 225); //Yellow
ppc.iSeconds = 60;
m_hPopupClass = Popup_RegisterClass(&ppc);
IcoLib_ReleaseIcon(ppc.hIcon);
}
void CTelegramProto::Popup(MCONTACT hContact, const wchar_t *szMsg, const wchar_t *szTitle)
{
if (!m_bUsePopups)
return;
char name[256];
mir_snprintf(name, "%s_%s", m_szModuleName, "Error");
CMStringW wszTitle(szTitle);
if (hContact == 0) {
wszTitle.Insert(0, L": ");
wszTitle.Insert(0, m_tszUserName);
}
POPUPDATACLASS ppd = {};
ppd.szTitle.w = wszTitle;
ppd.szText.w = szMsg;
ppd.pszClassName = name;
ppd.hContact = hContact;
Popup_AddClass(&ppd);
}
/////////////////////////////////////////////////////////////////////////////////////////
bool CTelegramProto::GetGcUserId(TG_USER *pUser, const TD::message *pMsg, char *dest)
{
if (pUser->isGroupChat) {
if (auto *pSender = GetSender(pMsg->sender_id_.get())) {
_i64toa(pSender->id, dest, 10);
CMStringW wszDisplayName(pSender->getDisplayName());
if (pUser->m_si && !pSender->wszFirstName.IsEmpty())
g_chatApi.UM_AddUser(pUser->m_si, Utf2T(dest), wszDisplayName, ID_STATUS_ONLINE);
mir_strncpy(dest, T2Utf(wszDisplayName), 100);
return true;
}
}
*dest = 0;
return false;
}
/////////////////////////////////////////////////////////////////////////////////////////
bool CTelegramProto::GetMessageFile(const EmbeddedFile &F, TG_FILE_REQUEST::Type iType, const TD::file *pFile, const char *pszFileName, const char *pszCaption)
{
auto *pRequest = new TG_FILE_REQUEST(iType, pFile->id_, pFile->remote_->id_.c_str());
pRequest->m_fileName = Utf2T(pszFileName);
pRequest->m_fileSize = pFile->size_;
pRequest->m_bRecv = !F.pMsg->is_outgoing_;
pRequest->m_hContact = GetRealContact(F.pUser);
if (mir_strlen(pszCaption))
F.szBody += pszCaption;
char szReplyId[100];
DB::EventInfo dbei(db_event_getById(m_szModuleName, F.pszId));
dbei.flags = DBEF_TEMPORARY;
dbei.timestamp = F.pMsg->date_;
dbei.szId = F.pszId;
dbei.szUserId = F.pszUser;
if (F.pMsg->is_outgoing_)
dbei.flags |= DBEF_SENT;
if (!F.pUser->bInited || F.bRead)
dbei.flags |= DBEF_READ;
if (auto iReplyId = getReplyId(F.pMsg->reply_to_.get())) {
_i64toa(iReplyId, szReplyId, 10);
dbei.szReplyId = szReplyId;
}
if (dbei) {
if (!Ignore_IsIgnored(pRequest->m_hContact, IGNOREEVENT_FILE)) {
DB::FILE_BLOB blob(dbei);
OnReceiveOfflineFile(dbei, blob);
blob.write(dbei);
db_event_edit(dbei.getEvent(), &dbei, true);
}
delete pRequest;
}
else ProtoChainRecvFile(pRequest->m_hContact, DB::FILE_BLOB(pRequest, pszFileName, F.szBody), dbei);
F.szBody.Empty();
return true;
}
CMStringA CTelegramProto::GetMessagePreview(const TD::file *pFile)
{
auto *pFileId = pFile->remote_->unique_id_.c_str();
auto *pRequest = new TG_FILE_REQUEST(TG_FILE_REQUEST::AVATAR, pFile->id_, pFileId);
pRequest->m_destPath = GetPreviewPath();
CreateDirectoryTreeW(pRequest->m_destPath);
pRequest->m_fileName.Format(L"{%S}.jpg", pFileId);
{
mir_cslock lck(m_csFiles);
m_arFiles.insert(pRequest);
}
SendQuery(new TD::downloadFile(pFile->id_, 10, 0, 0, true));
return T2Utf(pRequest->m_destPath + L"\\" + pRequest->m_fileName).get();
}
CMStringA CTelegramProto::GetMessageSticker(const TD::file *pFile, const char *pwszExtension)
{
auto *pFileId = pFile->remote_->unique_id_.c_str();
auto *pRequest = new TG_FILE_REQUEST(TG_FILE_REQUEST::AVATAR, pFile->id_, pFileId);
pRequest->m_isSmiley = true;
pRequest->m_destPath = GetAvatarPath() + L"\\Stickers";
CreateDirectoryW(pRequest->m_destPath, 0);
pRequest->m_fileName.Format(L"STK{%S}.%S", pFileId, pwszExtension);
{
mir_cslock lck(m_csFiles);
m_arFiles.insert(pRequest);
}
SendQuery(new TD::downloadFile(pFile->id_, 10, 0, 0, true));
return CMStringA(FORMAT, "STK{%s}", pFileId);
}
/////////////////////////////////////////////////////////////////////////////////////////
static const TD::photoSize* GetBiggestPhoto(const TD::photo *pPhoto)
{
const char *types[] = { "y", "x", "m", "s" };
for (auto *pType : types)
for (auto &it : pPhoto->sizes_)
if (it->type_ == pType)
return it.get();
return nullptr;
}
static bool checkStickerType(uint32_t ID)
{
switch (ID) {
case TD::stickerTypeRegular::ID:
case TD::stickerFullTypeRegular::ID:
return true;
default:
return false;
}
}
CMStringA CTelegramProto::GetMessageText(TG_USER *pUser, const TD::message *pMsg, bool bSkipJoin, bool bRead)
{
const TD::MessageContent *pBody = pMsg->content_.get();
char szUserId[100], *pszUserId = nullptr;
auto szMsgId(msg2id(pMsg));
if (GetGcUserId(pUser, pMsg, szUserId))
pszUserId = szUserId;
CMStringA ret;
if (auto *pForward = pMsg->forward_info_.get()) {
CMStringW wszNick;
switch (pForward->origin_->get_id()) {
case TD::messageOriginUser::ID:
if (auto *p = FindUser(((TD::messageOriginUser *)pForward->origin_.get())->sender_user_id_))
wszNick = p->getDisplayName();
break;
case TD::messageOriginChat::ID:
if (auto *p = FindChat(((TD::messageOriginChat *)pForward->origin_.get())->sender_chat_id_))
wszNick = p->getDisplayName();
break;
case TD::messageOriginHiddenUser::ID:
if (auto *p = (TD::messageOriginHiddenUser *)pForward->origin_.get())
wszNick = Utf2T(p->sender_name_.c_str());
break;
case TD::messageOriginChannel::ID:
if (auto *p = FindChat(((TD::messageOriginChannel *)pForward->origin_.get())->chat_id_))
wszNick = p->getDisplayName();
break;
default:
wszNick = TranslateT("Unknown");
}
wchar_t wszDate[100];
TimeZone_PrintTimeStamp(0, pForward->date_, L"d t", wszDate, _countof(wszDate), 0);
CMStringW wszForward(FORMAT, L">%s %s %s\r\n", wszDate, wszNick.c_str(), TranslateT("wrote"));
ret.Insert(0, T2Utf(wszForward));
}
EmbeddedFile embed(ret);
embed.pUser = pUser;
embed.bRead = bRead;
embed.pszId= szMsgId;
embed.pszUser = szUserId;
embed.pMsg = pMsg;
switch (pBody->get_id()) {
case TD::messageChatUpgradeTo::ID:
if (auto *pUgrade = (TD::messageChatUpgradeTo *)pBody) {
MCONTACT hContact = pUser->hContact;
m_arChats.remove(pUser);
m_arUsers.remove(pUser);
SetId(hContact, pUgrade->supergroup_id_);
pUser = new TG_USER(pUgrade->supergroup_id_, hContact, true);
m_arUsers.insert(pUser);
}
break;
case TD::messageChatAddMembers::ID:
if (!bSkipJoin)
if (auto *pDoc = (TD::messageChatAddMembers *)pBody)
for (auto &it : pDoc->member_user_ids_)
GcChangeMember(pUser, pszUserId, it, true);
break;
case TD::messageChatDeleteMember::ID:
if (!bSkipJoin)
if (auto *pDoc = (TD::messageChatDeleteMember *)pBody)
GcChangeMember(pUser, pszUserId, pDoc->user_id_, false);
break;
case TD::messageChatChangeTitle::ID:
if (auto *pDoc = (TD::messageChatChangeTitle *)pBody) {
if (pUser->m_si)
Chat_ChangeSessionName(pUser->m_si, Utf2T(pDoc->title_.c_str()));
else
setUString(pUser->hContact, "Nick", pDoc->title_.c_str());
ret.AppendFormat(TranslateU("Chat name was changed to %s"), pDoc->title_.c_str());
}
break;
case TD::messagePhoto::ID:
if (auto *pDoc = (TD::messagePhoto *)pBody) {
auto *pPhoto = GetBiggestPhoto(pDoc->photo_.get());
if (pPhoto == nullptr) {
debugLogA("cannot find photo, exiting");
break;
}
CMStringA fileName(FORMAT, "%s (%d x %d)", TranslateU("Picture"), pPhoto->width_, pPhoto->height_);
GetMessageFile(embed, TG_FILE_REQUEST::PICTURE, pPhoto->photo_.get(), fileName, pDoc->caption_->text_.c_str());
}
break;
case TD::messageAudio::ID:
if (auto *pDoc = (TD::messageAudio *)pBody) {
auto *pAudio = pDoc->audio_.get();
CMStringA caption(FORMAT, "%s (%d %s)", TranslateU("Audio"), pAudio->duration_, TranslateU("seconds"));
if (!pDoc->caption_->text_.empty()) {
caption += " ";
caption += pDoc->caption_->text_.c_str();
}
GetMessageFile(embed, TG_FILE_REQUEST::VIDEO, pAudio->audio_.get(), pAudio->file_name_.c_str(), caption);
}
break;
case TD::messageVideo::ID:
if (auto *pDoc = (TD::messageVideo *)pBody) {
auto *pVideo = pDoc->video_.get();
CMStringA caption(FORMAT, "%s (%d x %d, %d %s)", TranslateU("Video"), pVideo->width_, pVideo->height_, pVideo->duration_, TranslateU("seconds"));
if (!pDoc->caption_->text_.empty()) {
caption += " ";
caption += pDoc->caption_->text_.c_str();
}
GetMessageFile(embed, TG_FILE_REQUEST::VIDEO, pVideo->video_.get(), pVideo->file_name_.c_str(), caption);
}
break;
case TD::messageAnimation::ID:
if (auto *pDoc = (TD::messageAnimation *)pBody) {
auto *pVideo = pDoc->animation_.get();
CMStringA caption(FORMAT, "%s (%d x %d, %d %s)", TranslateU("Video"), pVideo->width_, pVideo->height_, pVideo->duration_, TranslateU("seconds"));
if (!pDoc->caption_->text_.empty()) {
caption += " ";
caption += pDoc->caption_->text_.c_str();
}
GetMessageFile(embed, TG_FILE_REQUEST::VIDEO, pVideo->animation_.get(), pVideo->file_name_.c_str(), caption.c_str());
}
break;
case TD::messageVideoNote::ID:
if (auto *pDoc = (TD::messageVideoNote *)pBody) {
auto *pVideo = pDoc->video_note_.get();
CMStringA fileName(FORMAT, "%s (%d %s)", TranslateU("Video note"), pVideo->duration_, TranslateU("seconds"));
GetMessageFile(embed, TG_FILE_REQUEST::VIDEO, pVideo->video_.get(), fileName, "");
}
break;
case TD::messageVoiceNote::ID:
if (auto *pDoc = (TD::messageVoiceNote *)pBody) {
CMStringA fileName(FORMAT, "%s (%d %s)", TranslateU("Voice message"), pDoc->voice_note_->duration_, TranslateU("seconds"));
GetMessageFile(embed, TG_FILE_REQUEST::VOICE, pDoc->voice_note_->voice_.get(), fileName, pDoc->caption_->text_.c_str());
}
break;
case TD::messageDocument::ID:
if (auto *pDoc = (TD::messageDocument *)pBody)
GetMessageFile(embed, TG_FILE_REQUEST::FILE, pDoc->document_->document_.get(), pDoc->document_->file_name_.c_str(), pDoc->caption_->text_.c_str());
break;
case TD::messageAnimatedEmoji::ID:
if (auto *pObj = (TD::messageAnimatedEmoji *)pBody) {
if (m_bSmileyAdd) {
if (auto *pAnimated = pObj->animated_emoji_.get()) {
if (auto *pSticker = pAnimated->sticker_.get()) {
if (!checkStickerType(pSticker->full_type_->get_id())) {
debugLogA("You received a sticker of unsupported type %d, ignored", pSticker->full_type_->get_id());
break;
}
const char *pwszFileExt;
switch (pSticker->thumbnail_->format_->get_id()) {
case TD::thumbnailFormatGif::ID: pwszFileExt = "gif"; break;
case TD::thumbnailFormatPng::ID: pwszFileExt = "png"; break;
case TD::thumbnailFormatTgs::ID: pwszFileExt = "tga"; break;
case TD::thumbnailFormatJpeg::ID: pwszFileExt = "jpg"; break;
case TD::thumbnailFormatWebm::ID: pwszFileExt = "webm"; break;
case TD::thumbnailFormatWebp::ID: pwszFileExt = "webp"; break;
default:pwszFileExt = "jpeg"; break;
}
ret = GetMessageSticker(pSticker->thumbnail_->file_.get(), pwszFileExt);
break;
}
}
}
ret += pObj->emoji_.c_str();
}
break;
case TD::messageSticker::ID:
if (auto *pSticker = ((TD::messageSticker *)pBody)->sticker_.get()) {
if (!checkStickerType(pSticker->full_type_->get_id())) {
debugLogA("You received a sticker of unsupported type %d, ignored", pSticker->full_type_->get_id());
break;
}
if (m_bSmileyAdd) {
if (pSticker->thumbnail_.get()) {
const char *pwszFileExt;
switch (pSticker->format_->get_id()) {
case TD::stickerFormatTgs::ID: pwszFileExt = "tga"; break;
case TD::stickerFormatWebm::ID: pwszFileExt = "webm"; break;
case TD::stickerFormatWebp::ID: pwszFileExt = "webp"; break;
default:
pwszFileExt = "jpeg"; break;
}
ret = GetMessageSticker(pSticker->thumbnail_->file_.get(), pwszFileExt);
}
else {
debugLogA("Strange sticker without preview");
ret.Append(pSticker->emoji_.c_str());
}
}
else ret.AppendFormat("%s: %s", TranslateU("SmileyAdd plugin required to support stickers"), pSticker->emoji_.c_str());
}
break;
case TD::messageInvoice::ID:
if (auto *pInvoice = ((TD::messageInvoice *)pBody)) {
ret.Format("%s: %.2lf %s", TranslateU("You received an invoice"), double(pInvoice->total_amount_)/100.0, pInvoice->currency_.c_str());
if (auto pszText = GetFormattedText(pInvoice->paid_media_caption_))
ret.AppendFormat("\r\n%s", pszText.c_str());
}
break;
case TD::messageText::ID:
if (auto *pText = ((TD::messageText *)pBody)) {
ret = GetFormattedText(pText->text_);
auto *pWeb = pText->link_preview_.get();
if (pWeb && m_bIncludePreviews) {
CMStringA szDescr;
if (!pWeb->site_name_.empty())
szDescr.AppendFormat("%s\r\n", pWeb->site_name_.c_str());
if (!pWeb->title_.empty())
szDescr.AppendFormat("[b]%s[/b]\r\n", pWeb->title_.c_str());
if (auto szText = GetFormattedText(pWeb->description_))
szDescr.AppendFormat("%s\r\n", szText.c_str());
if (pWeb->type_->get_id() == TD::linkPreviewTypePhoto::ID) {
auto *pPhoto = ((TD::linkPreviewTypePhoto *)pWeb->type_.get())->photo_.get();
const TD::photoSize *pSize = nullptr;
for (auto &it : pPhoto->sizes_)
if (it->type_ == "m")
pSize = it.get();
if (pSize == nullptr)
pSize = pPhoto->sizes_[0].get();
if (auto szText = GetMessagePreview(pSize->photo_.get()))
szDescr.AppendFormat("[img=%s][/img]\r\n", szText.c_str());
}
if (!szDescr.IsEmpty())
ret += "\r\n[quote]" + szDescr.Trim() + "[/quote]";
}
}
break;
}
return ret;
}