/* 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" struct { const wchar_t *begin, *end; unsigned len1, len2; } static bbCodes[] = { { L"[b]", L"[/b]", 3, 4 }, { L"[i]", L"[/i]", 3, 4 }, { L"[s]", L"[/s]", 3, 4 }, { L"[u]", L"[/u]", 3, 4 }, }; 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.begin[1]) { case 'b': pNew = TD::make_object(); break; case 'i': pNew = TD::make_object(); break; case 's': pNew = TD::make_object(); break; case 'u': 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; } static CMStringA getFormattedText(TD::object_ptr &pText) { if (pText->get_id() == TD::formattedText::ID) { 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; 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(); } return ""; } ///////////////////////////////////////////////////////////////////////////////////////// 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; } CMStringW TG_USER::getDisplayName() const { if (hContact != INVALID_CONTACT_ID) return Clist_GetContactDisplayName(hContact, 0); if (!wszFirstName.IsEmpty()) return (wszLastName.IsEmpty()) ? wszFirstName : wszFirstName + L" " + wszLastName; return wszNick; } 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) { return _atoi64(getMStringA(hContact, DBKEY_ID)); } void CTelegramProto::SetId(MCONTACT hContact, int64_t id) { char szId[100]; _i64toa(id, szId, 10); setString(hContact, DBKEY_ID, 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); if (pUser->m_si && !pSender->wszFirstName.IsEmpty()) g_chatApi.UM_AddUser(pUser->m_si, Utf2T(dest), pSender->getDisplayName(), ID_STATUS_ONLINE); return true; } } *dest = 0; return false; } ///////////////////////////////////////////////////////////////////////////////////////// bool CTelegramProto::GetMessageFile( TG_FILE_REQUEST::Type fileType, TG_USER *pUser, const TD::file *pFile, const char *pszFileName, const std::string &caption, const char *pszId, const char *pszUserId, const TD::message *pMsg) { if (pFile->get_id() != TD::file::ID) { debugLogA("Document contains unsupported type %d, exiting", pFile->get_id()); return false; } auto *pRequest = new TG_FILE_REQUEST(fileType, pFile->id_, pFile->remote_->id_.c_str()); pRequest->m_fileName = Utf2T(pszFileName); pRequest->m_fileSize = pFile->size_; pRequest->m_bRecv = true; { mir_cslock lck(m_csFiles); m_arFiles.insert(pRequest); } char szReplyId[100]; const char *szDesc = nullptr; MCONTACT hContact = GetRealContact(pUser); DB::EventInfo dbei; dbei.flags = DBEF_TEMPORARY; dbei.timestamp = pMsg->date_; dbei.szId = pszId; dbei.szUserId = pszUserId; if (!caption.empty()) szDesc = caption.c_str(); if (pMsg->is_outgoing_) dbei.flags |= DBEF_SENT; if (Contact::IsGroupChat(hContact) || !pUser->bInited) dbei.flags |= DBEF_READ; if (pMsg->reply_to_message_id_) { _i64toa(pMsg->reply_to_message_id_, szReplyId, 10); dbei.szReplyId = szReplyId; } ProtoChainRecvFile(hContact, DB::FILE_BLOB(pRequest, pszFileName, szDesc), dbei); return true; } 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_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) { const TD::MessageContent *pBody = pMsg->content_.get(); char szUserId[100], *pszUserId = nullptr; auto szMsgId(msg2id(pMsg)); if (GetGcUserId(pUser, pMsg, szUserId)) pszUserId = szUserId; 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) GcChangeTopic(pUser, Utf2T(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(TG_FILE_REQUEST::PICTURE, pUser, pPhoto->photo_.get(), fileName, pDoc->caption_->text_, szMsgId, pszUserId, pMsg); } break; case TD::messageAudio::ID: if (auto *pDoc = (TD::messageAudio *)pBody) { auto *pAudio = pDoc->audio_.get(); CMStringA fileName(FORMAT, "%s (%d %s)", TranslateU("Audio"), pAudio->duration_, TranslateU("seconds")); std::string caption = fileName.c_str(); if (!pDoc->caption_->text_.empty()) { caption += " "; caption += pDoc->caption_->text_; } GetMessageFile(TG_FILE_REQUEST::VIDEO, pUser, pAudio->audio_.get(), pAudio->file_name_.c_str(), caption, szMsgId, pszUserId, pMsg); } break; case TD::messageVideo::ID: if (auto *pDoc = (TD::messageVideo *)pBody) { auto *pVideo = pDoc->video_.get(); CMStringA fileName(FORMAT, "%s (%d x %d, %d %s)", TranslateU("Video"), pVideo->width_, pVideo->height_, pVideo->duration_, TranslateU("seconds")); std::string caption = fileName.c_str(); if (!pDoc->caption_->text_.empty()) { caption += " "; caption += pDoc->caption_->text_; } GetMessageFile(TG_FILE_REQUEST::VIDEO, pUser, pVideo->video_.get(), pVideo->file_name_.c_str(), caption, szMsgId, pszUserId, pMsg); } break; case TD::messageAnimation::ID: if (auto *pDoc = (TD::messageAnimation *)pBody) { auto *pVideo = pDoc->animation_.get(); CMStringA fileName(FORMAT, "%s (%d x %d, %d %s)", TranslateU("Video"), pVideo->width_, pVideo->height_, pVideo->duration_, TranslateU("seconds")); std::string caption = fileName.c_str(); if (!pDoc->caption_->text_.empty()) { caption += " "; caption += pDoc->caption_->text_; } GetMessageFile(TG_FILE_REQUEST::VIDEO, pUser, pVideo->animation_.get(), pVideo->file_name_.c_str(), caption, szMsgId, pszUserId, pMsg); } 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(TG_FILE_REQUEST::VOICE, pUser, pDoc->voice_note_->voice_.get(), fileName, pDoc->caption_->text_, szMsgId, pszUserId, pMsg); } break; case TD::messageDocument::ID: if (auto *pDoc = (TD::messageDocument *)pBody) GetMessageFile(TG_FILE_REQUEST::FILE, pUser, pDoc->document_->document_.get(), pDoc->document_->file_name_.c_str(), pDoc->caption_->text_, szMsgId, pszUserId, pMsg); 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; } return GetMessageSticker(pSticker->thumbnail_->file_.get(), pwszFileExt); } } } return pObj->emoji_.c_str(); } 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) return CMStringA(FORMAT, "%s: %s", TranslateU("SmileyAdd plugin required to support stickers"), pSticker->emoji_.c_str()); 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; } return GetMessageSticker(pSticker->thumbnail_->file_.get(), pwszFileExt); } case TD::messageInvoice::ID: { auto *pInvoice = ((TD::messageInvoice *)pBody); CMStringA ret(FORMAT, "%s: %.2lf %s", TranslateU("You received an invoice"), double(pInvoice->total_amount_)/100.0, pInvoice->currency_.c_str()); if (!pInvoice->title_.empty()) ret.AppendFormat("\r\n%s: %s", TranslateU("Title"), pInvoice->title_.c_str()); if (auto pszText = getFormattedText(pInvoice->description_)) ret.AppendFormat("\r\n%s", pszText.c_str()); return ret; } case TD::messageText::ID: if (auto pszText = getFormattedText(((TD::messageText *)pBody)->text_)) return pszText; break; } return CMStringA(); }