0.8.25.dev version with ipv6, bots profiles, keyboard and command autocomplete + elided text align fixed

This commit is contained in:
John Preston 2015-06-15 20:19:24 +03:00
parent 83744e77d1
commit 84c2a33c18
42 changed files with 1667 additions and 354 deletions

View File

@ -1,10 +1,10 @@
@echo OFF @echo OFF
set "AppVersion=8024" set "AppVersion=8025"
set "AppVersionStrSmall=0.8.24" set "AppVersionStrSmall=0.8.25"
set "AppVersionStr=0.8.24" set "AppVersionStr=0.8.25"
set "AppVersionStrFull=0.8.24.0" set "AppVersionStrFull=0.8.25.0"
set "DevChannel=0" set "DevChannel=1"
if %DevChannel% neq 0 goto preparedev if %DevChannel% neq 0 goto preparedev

View File

@ -70,6 +70,8 @@ Copyright (c) 2014 John Preston, https://desktop.telegram.org
"lng_status_service_notifications" = "service notifications"; "lng_status_service_notifications" = "service notifications";
"lng_status_bot" = "bot"; "lng_status_bot" = "bot";
"lng_status_bot_reads_all" = "sees all messages";
"lng_status_bot_not_reads_all" = "only sees messages starting with /";
"lng_status_offline" = "last seen a long time ago"; "lng_status_offline" = "last seen a long time ago";
"lng_status_recently" = "last seen recently"; "lng_status_recently" = "last seen recently";
"lng_status_last_week" = "last seen within a week"; "lng_status_last_week" = "last seen within a week";
@ -334,7 +336,10 @@ Copyright (c) 2014 John Preston, https://desktop.telegram.org
"lng_profile_chat_unaccessible" = "Group is unaccessible"; "lng_profile_chat_unaccessible" = "Group is unaccessible";
"lng_topbar_info" = "Info"; "lng_topbar_info" = "Info";
"lng_profile_about_section" = "About";
"lng_profile_settings_section" = "Settings"; "lng_profile_settings_section" = "Settings";
"lng_profile_bot_settings" = "Settings";
"lng_profile_bot_help" = "Help";
"lng_profile_participants_section" = "Members"; "lng_profile_participants_section" = "Members";
"lng_profile_info" = "Contact info"; "lng_profile_info" = "Contact info";
"lng_profile_group_info" = "Group info"; "lng_profile_group_info" = "Group info";
@ -344,6 +349,7 @@ Copyright (c) 2014 John Preston, https://desktop.telegram.org
"lng_profile_clear_history" = "Clear history"; "lng_profile_clear_history" = "Clear history";
"lng_profile_send_message" = "Send Message"; "lng_profile_send_message" = "Send Message";
"lng_profile_share_contact" = "Share Contact"; "lng_profile_share_contact" = "Share Contact";
"lng_profile_invite_to_group" = "Add to Group";
"lng_profile_delete_contact" = "Delete"; "lng_profile_delete_contact" = "Delete";
"lng_profile_set_group_photo" = "Set Photo"; "lng_profile_set_group_photo" = "Set Photo";
"lng_profile_add_participant" = "Add Member"; "lng_profile_add_participant" = "Add Member";
@ -458,6 +464,12 @@ Copyright (c) 2014 John Preston, https://desktop.telegram.org
"lng_from_you" = "You"; "lng_from_you" = "You";
"lng_bot_description" = "What can this bot do?"; "lng_bot_description" = "What can this bot do?";
"lng_bot_choose_group" = "Choose Group";
"lng_bot_no_groups" = "You have no groups";
"lng_bot_groups_not_found" = "No groups found";
"lng_bot_sure_invite" = "Add the bot to «{group}»?";
"lng_bot_already_in_group" = "The bot is already a member of the group.";
"lng_typing" = "typing"; "lng_typing" = "typing";
"lng_user_typing" = "{user} is typing"; "lng_user_typing" = "{user} is typing";
"lng_users_typing" = "{user} and {second_user} are typing"; "lng_users_typing" = "{user} and {second_user} are typing";
@ -587,7 +599,7 @@ Copyright (c) 2014 John Preston, https://desktop.telegram.org
"lng_new_version_wrap" = "Telegram Desktop was updated to version {version}\n\n{changes}\n\nFull version history is available here:\n{link}"; "lng_new_version_wrap" = "Telegram Desktop was updated to version {version}\n\n{changes}\n\nFull version history is available here:\n{link}";
"lng_new_version_minor" = "— Bug fixes and other minor improvements"; "lng_new_version_minor" = "— Bug fixes and other minor improvements";
"lng_new_version_text" = "— Improved sticker panel\n— Bug fixes and minor stuff"; "lng_new_version_text" = "This new version includes support for bots using the new bot API. If you're an engineer, create your own bots like @quiz_bot or @hot_or_bot using the @botfather.\n\nLearn more at {blog_link}";
"lng_menu_insert_unicode" = "Insert Unicode control character"; "lng_menu_insert_unicode" = "Insert Unicode control character";

View File

@ -992,6 +992,18 @@ btnAttachEmoji: iconedButton(btnAttachDocument) {
width: 33px; width: 33px;
} }
btnBotKbShow: iconedButton(btnAttachEmoji) {
icon: sprite(375px, 74px, 21px, 16px);
iconPos: point(6px, 16px);
downIcon: sprite(375px, 74px, 21px, 16px);
downIconPos: point(6px, 16px);
}
btnBotKbHide: iconedButton(btnAttachEmoji) {
icon: sprite(352px, 74px, 23px, 14px);
iconPos: point(5px, 17px);
downIcon: sprite(352px, 74px, 23px, 14px);
downIconPos: point(5px, 17px);
}
btnRecordAudio: sprite(363px, 366px, 16px, 24px); btnRecordAudio: sprite(363px, 366px, 16px, 24px);
btnRecordAudioActive: sprite(379px, 366px, 16px, 24px); btnRecordAudioActive: sprite(379px, 366px, 16px, 24px);
recordSignalColor: #f17077; recordSignalColor: #f17077;
@ -1041,7 +1053,7 @@ textRectMargins: margins(-2px, -1px, -2px, -1px);
taMsgField: flatTextarea(taDefFlat) { taMsgField: flatTextarea(taDefFlat) {
font: msgFont; font: msgFont;
} }
maxFieldHeight: 250px; maxFieldHeight: 265px;
newMsgSound: ':/gui/art/newmsg.wav'; newMsgSound: ':/gui/art/newmsg.wav';
@ -1658,6 +1670,31 @@ stickerIconLeft: sprite(342px, 72px, 40px, 1px);
stickerIconRight: sprite(342px, 73px, 40px, 1px); stickerIconRight: sprite(342px, 73px, 40px, 1px);
stickerIconMove: 400; stickerIconMove: 400;
botKbDuration: 200;
botKbBg: #f7f7f7;
botKbOverBg: #e8ecef;
botKbDownBg: #dfe3e6;
botKbColor: #8a8a8f;
botKbFont: font(16px);
botKbButton: botKeyboardButton {
margin: 10px;
padding: 14px;
height: 46px;
textTop: 13px;
downTextTop: 14px;
}
botKbTinyButton: botKeyboardButton {
margin: 4px;
padding: 3px;
height: 25px;
textTop: 2px;
downTextTop: 3px;
}
botKbScroll: flatScroll(newScroll) {
deltax: 3px;
width: 10px;
}
mvBgColor: #222; mvBgColor: #222;
mvBgOpacity: 0.92; mvBgOpacity: 0.92;
mvThickFont: font(fsize semibold); mvThickFont: font(fsize semibold);
@ -1887,6 +1924,8 @@ mentionPadding: margins(8px, 5px, 8px, 5px);
mentionTop: 11px; mentionTop: 11px;
mentionFont: linkFont; mentionFont: linkFont;
mentionPhotoSize: msgPhotoSize; mentionPhotoSize: msgPhotoSize;
botCommandFont: font(fsize semibold);
botDescFont: font(fsize italic);
sessionsHeight: 440px; sessionsHeight: 440px;
sessionHeight: 70px; sessionHeight: 70px;

View File

@ -259,3 +259,11 @@ dropdown {
duration: number; duration: number;
width: number; width: number;
} }
botKeyboardButton {
margin: number;
padding: number;
height: number;
textTop: number;
downTextTop: number;
}

View File

@ -119,6 +119,19 @@ void ApiWrap::gotChatFull(PeerData *peer, const MTPmessages_ChatFull &result) {
App::feedUsers(d.vusers); App::feedUsers(d.vusers);
App::feedChats(d.vchats); App::feedChats(d.vchats);
App::feedParticipants(f.vparticipants); App::feedParticipants(f.vparticipants);
const QVector<MTPBotInfo> &v(f.vbot_info.c_vector().v);
for (QVector<MTPBotInfo>::const_iterator i = v.cbegin(), e = v.cend(); i < e; ++i) {
switch (i->type()) {
case mtpc_botInfo: {
const MTPDbotInfo &b(i->c_botInfo());
UserData *user = App::userLoaded(b.vuser_id.v);
if (user) {
user->setBotInfo(*i);
emit fullPeerUpdated(user);
}
} break;
}
}
PhotoData *photo = App::feedPhoto(f.vchat_photo); PhotoData *photo = App::feedPhoto(f.vchat_photo);
ChatData *chat = peer->asChat(); ChatData *chat = peer->asChat();
if (chat) { if (chat) {
@ -132,7 +145,7 @@ void ApiWrap::gotChatFull(PeerData *peer, const MTPmessages_ChatFull &result) {
App::main()->gotNotifySetting(MTP_inputNotifyPeer(peer->input), f.vnotify_settings); App::main()->gotNotifySetting(MTP_inputNotifyPeer(peer->input), f.vnotify_settings);
_fullRequests.remove(peer); _fullRequests.remove(peer);
emit fullPeerLoaded(peer); emit fullPeerUpdated(peer);
} }
void ApiWrap::gotUserFull(PeerData *peer, const MTPUserFull &result) { void ApiWrap::gotUserFull(PeerData *peer, const MTPUserFull &result) {
@ -145,7 +158,7 @@ void ApiWrap::gotUserFull(PeerData *peer, const MTPUserFull &result) {
peer->asUser()->setBotInfo(d.vbot_info); peer->asUser()->setBotInfo(d.vbot_info);
_fullRequests.remove(peer); _fullRequests.remove(peer);
emit fullPeerLoaded(peer); emit fullPeerUpdated(peer);
} }
bool ApiWrap::gotPeerFailed(PeerData *peer, const RPCError &error) { bool ApiWrap::gotPeerFailed(PeerData *peer, const RPCError &error) {

View File

@ -40,7 +40,7 @@ public:
signals: signals:
void fullPeerLoaded(PeerData *peer); void fullPeerUpdated(PeerData *peer);
public slots: public slots:

View File

@ -55,6 +55,9 @@ namespace {
typedef QHash<WebPageId, WebPageData*> WebPagesData; typedef QHash<WebPageId, WebPageData*> WebPagesData;
WebPagesData webPagesData; WebPagesData webPagesData;
typedef QMap<MsgId, ReplyMarkup> ReplyMarkups;
ReplyMarkups replyMarkups;
VideoItems videoItems; VideoItems videoItems;
AudioItems audioItems; AudioItems audioItems;
DocumentItems documentItems; DocumentItems documentItems;
@ -208,7 +211,7 @@ namespace App {
int32 onlineForSort(UserData *user, int32 now) { int32 onlineForSort(UserData *user, int32 now) {
if (isServiceUser(user->id) || user->botInfo) { if (isServiceUser(user->id) || user->botInfo) {
return now - 1; return -1;
} }
int32 online = user->onlineTill; int32 online = user->onlineTill;
if (online <= 0) { if (online <= 0) {
@ -343,6 +346,7 @@ namespace App {
data->setName(lang(lng_deleted), QString(), QString(), QString()); data->setName(lang(lng_deleted), QString(), QString(), QString());
data->setPhoto(MTP_userProfilePhotoEmpty()); data->setPhoto(MTP_userProfilePhotoEmpty());
data->access = UserNoAccess; data->access = UserNoAccess;
data->setBotInfoVersion(-1);
wasContact = (data->contact > 0); wasContact = (data->contact > 0);
status = &emptyStatus; status = &emptyStatus;
data->contact = -1; data->contact = -1;
@ -382,7 +386,13 @@ namespace App {
status = d.has_status() ? &d.vstatus : &emptyStatus; status = d.has_status() ? &d.vstatus : &emptyStatus;
} }
wasContact = (data->contact > 0); wasContact = (data->contact > 0);
if (d.has_bot_info_version()) data->setBotInfoVersion(d.vbot_info_version.v); if (d.has_bot_info_version()) {
data->setBotInfoVersion(d.vbot_info_version.v);
data->botInfo->readsAllHistory = (d.vflags.v & MTPDuser_flag_bot_reads_all);
data->botInfo->cantJoinGroups = (d.vflags.v & MTPDuser_flag_bot_cant_join);
} else {
data->setBotInfoVersion(-1);
}
data->contact = (flags & (MTPDuser_flag_contact | MTPDuser_flag_mutual_contact)) ? 1 : (data->phone.isEmpty() ? -1 : 0); data->contact = (flags & (MTPDuser_flag_contact | MTPDuser_flag_mutual_contact)) ? 1 : (data->phone.isEmpty() ? -1 : 0);
if ((flags & MTPDuser_flag_self) && ::self != data) { if ((flags & MTPDuser_flag_self) && ::self != data) {
::self = data; ::self = data;
@ -446,6 +456,7 @@ namespace App {
if (data->version < d.vversion.v) { if (data->version < d.vversion.v) {
data->version = d.vversion.v; data->version = d.vversion.v;
data->participants = ChatData::Participants(); data->participants = ChatData::Participants();
data->botStatus = 0;
} }
} break; } break;
case mtpc_chatForbidden: { case mtpc_chatForbidden: {
@ -479,6 +490,7 @@ namespace App {
if (data->version < d.vversion.v) { if (data->version < d.vversion.v) {
data->version = d.vversion.v; data->version = d.vversion.v;
data->participants = ChatData::Participants(); data->participants = ChatData::Participants();
data->botStatus = 0;
}/**/ }/**/
} break; } break;
} }
@ -492,7 +504,7 @@ namespace App {
return data; return data;
} }
void feedParticipants(const MTPChatParticipants &p) { void feedParticipants(const MTPChatParticipants &p, bool requestBotInfos) {
switch (p.type()) { switch (p.type()) {
case mtpc_chatParticipantsForbidden: { case mtpc_chatParticipantsForbidden: {
const MTPDchatParticipantsForbidden &d(p.c_chatParticipantsForbidden()); const MTPDchatParticipantsForbidden &d(p.c_chatParticipantsForbidden());
@ -519,17 +531,24 @@ namespace App {
} }
} else { } else {
chat->participants = ChatData::Participants(); chat->participants = ChatData::Participants();
chat->botStatus = 0;
break; break;
} }
} }
if (!chat->participants.isEmpty()) { if (!chat->participants.isEmpty()) {
int32 botStatus = -1;
for (ChatData::Participants::iterator i = chat->participants.begin(), e = chat->participants.end(); i != e;) { for (ChatData::Participants::iterator i = chat->participants.begin(), e = chat->participants.end(); i != e;) {
if (i.value() < pversion) { if (i.value() < pversion) {
i = chat->participants.erase(i); i = chat->participants.erase(i);
} else { } else {
if (i.key()->botInfo) {
botStatus = (botStatus > 0 || i.key()->botInfo->readsAllHistory) ? 2 : 1;
if (requestBotInfos && !i.key()->botInfo->inited) App::api()->requestFullPeer(i.key());
}
++i; ++i;
} }
} }
chat->botStatus = botStatus;
} }
if (App::main()) App::main()->peerUpdated(chat); if (App::main()) App::main()->peerUpdated(chat);
} }
@ -545,6 +564,7 @@ namespace App {
if (user) { if (user) {
if (chat->participants.isEmpty() && chat->count) { if (chat->participants.isEmpty() && chat->count) {
chat->count++; chat->count++;
chat->botStatus = 0;
} else if (chat->participants.find(user) == chat->participants.end()) { } else if (chat->participants.find(user) == chat->participants.end()) {
chat->participants[user] = (chat->participants.isEmpty() ? 1 : chat->participants.begin().value()); chat->participants[user] = (chat->participants.isEmpty() ? 1 : chat->participants.begin().value());
if (d.vinviter_id.v == MTP::authedId()) { if (d.vinviter_id.v == MTP::authedId()) {
@ -553,9 +573,14 @@ namespace App {
chat->cankick.remove(user); chat->cankick.remove(user);
} }
chat->count++; chat->count++;
if (user->botInfo) {
chat->botStatus = (chat->botStatus > 0 || !user->botInfo->readsAllHistory) ? 2 : 1;
if (!user->botInfo->inited) App::api()->requestFullPeer(user);
}
} }
} else { } else {
chat->participants = ChatData::Participants(); chat->participants = ChatData::Participants();
chat->botStatus = 0;
chat->count++; chat->count++;
} }
if (App::main()) App::main()->peerUpdated(chat); if (App::main()) App::main()->peerUpdated(chat);
@ -576,9 +601,23 @@ namespace App {
chat->participants.erase(i); chat->participants.erase(i);
chat->count--; chat->count--;
} }
if (chat->botStatus > 0 && user->botInfo) {
int32 botStatus = -1;
for (ChatData::Participants::const_iterator j = chat->participants.cbegin(), e = chat->participants.cend(); j != e; ++j) {
if (j.key()->botInfo) {
if (botStatus > 0 || !j.key()->botInfo->readsAllHistory) {
botStatus = 2;
break;
}
botStatus = 1;
}
}
chat->botStatus = botStatus;
}
} }
} else { } else {
chat->participants = ChatData::Participants(); chat->participants = ChatData::Participants();
chat->botStatus = 0;
chat->count--; chat->count--;
} }
if (App::main()) App::main()->peerUpdated(chat); if (App::main()) App::main()->peerUpdated(chat);
@ -1479,6 +1518,7 @@ namespace App {
} }
::maxMsgId = 0; ::maxMsgId = 0;
::hoveredItem = ::pressedItem = ::hoveredLinkItem = ::pressedLinkItem = ::contextItem = 0; ::hoveredItem = ::pressedItem = ::hoveredLinkItem = ::pressedLinkItem = ::contextItem = 0;
replyMarkups.clear();
} }
void historyClearItems() { void historyClearItems() {
@ -1634,6 +1674,9 @@ namespace App {
prepareCorners(MediaviewSaveCorners, st::msgRadius, st::emojiPanHover); prepareCorners(MediaviewSaveCorners, st::msgRadius, st::emojiPanHover);
prepareCorners(EmojiHoverCorners, st::msgRadius, st::emojiPanHover); prepareCorners(EmojiHoverCorners, st::msgRadius, st::emojiPanHover);
prepareCorners(StickerHoverCorners, st::msgRadius, st::emojiPanHover); prepareCorners(StickerHoverCorners, st::msgRadius, st::emojiPanHover);
prepareCorners(BotKeyboardCorners, st::msgRadius, st::botKbBg);
prepareCorners(BotKeyboardOverCorners, st::msgRadius, st::botKbOverBg);
prepareCorners(BotKeyboardDownCorners, st::msgRadius, st::botKbDownBg);
prepareCorners(MessageInCorners, st::msgRadius, st::msgInBg, &st::msgInShadow); prepareCorners(MessageInCorners, st::msgRadius, st::msgInBg, &st::msgInShadow);
prepareCorners(MessageInSelectedCorners, st::msgRadius, st::msgInSelectBg, &st::msgInSelectShadow); prepareCorners(MessageInSelectedCorners, st::msgRadius, st::msgInSelectBg, &st::msgInSelectShadow);
@ -1928,6 +1971,106 @@ namespace App {
if (changeInMin) App::main()->updateMutedIn(changeInMin); if (changeInMin) App::main()->updateMutedIn(changeInMin);
} }
void feedReplyMarkup(MsgId msgId, const MTPReplyMarkup &markup) {
ReplyMarkup data;
switch (markup.type()) {
case mtpc_replyKeyboardMarkup: {
const MTPDreplyKeyboardMarkup &d(markup.c_replyKeyboardMarkup());
const QVector<MTPKeyboardButtonRow> &v(d.vrows.c_vector().v);
if (!v.isEmpty()) {
data.reserve(v.size());
for (int32 i = 0, l = v.size(); i < l; ++i) {
switch (v.at(i).type()) {
case mtpc_keyboardButtonRow: {
const MTPDkeyboardButtonRow &r(v.at(i).c_keyboardButtonRow());
const QVector<MTPKeyboardButton> &b(r.vbuttons.c_vector().v);
if (!b.isEmpty()) {
QList<QString> btns;
btns.reserve(b.size());
for (int32 j = 0, s = b.size(); j < s; ++j) {
switch (b.at(j).type()) {
case mtpc_keyboardButton: {
btns.push_back(qs(b.at(j).c_keyboardButton().vtext));
} break;
}
}
if (!btns.isEmpty()) data.push_back(btns);
}
} break;
}
}
if (!data.isEmpty()) {
replyMarkups.insert(msgId, data);
}
}
} break;
}
}
void clearReplyMarkup(MsgId msgId) {
replyMarkups.remove(msgId);
}
const ReplyMarkup &replyMarkup(MsgId msgId) {
static ReplyMarkup zeroMarkup;
if (zeroMarkup.isEmpty()) {
QList<QString> cmds;
cmds.push_back("Test command 1Test comma");
cmds.push_back("Test comma" + emojiGetSequence(0));
zeroMarkup.push_back(cmds);
cmds.clear();
cmds.push_back("123 Test command 1");
cmds.push_back("321 Test command 3");
cmds.push_back("123 Test command 4");
zeroMarkup.push_back(cmds);
cmds.clear();
cmds.push_back("Test command 11111");
cmds.push_back("Test command 222222");
cmds.push_back("Test command 33333");
cmds.push_back("Test command 444444");
cmds.push_back("Test command 55555");
zeroMarkup.push_back(cmds);
cmds.clear();
cmds.push_back("123 1");
cmds.push_back("321 3");
zeroMarkup.push_back(cmds);
cmds.clear();
cmds.push_back("Test command 11111");
cmds.push_back("Test command 222222");
cmds.push_back("Test command 33333");
cmds.push_back("Test command 444444");
cmds.push_back("Test command 55555");
cmds.push_back("123 Test command 1");
cmds.push_back("321 Test command 3");
cmds.push_back("123 Test command 4");
zeroMarkup.push_back(cmds);
cmds.clear();
cmds.push_back("Test command 11111");
cmds.push_back("Test command 222222");
cmds.push_back("Test command 33333");
cmds.push_back("Test command 444444");
cmds.push_back("Test command 55555");
cmds.push_back("123 Test command 1");
cmds.push_back("321 Test command 3");
cmds.push_back("123 Test command 4");
zeroMarkup.push_back(cmds);
cmds.clear();
cmds.push_back("Test command 11111");
cmds.push_back("Test command 222222");
cmds.push_back("Test command 33333");
cmds.push_back("Test command 444444");
cmds.push_back("Test command 55555");
cmds.push_back("123 Test command 1");
cmds.push_back("321 Test command 3");
cmds.push_back("123 Test command 4");
zeroMarkup.push_back(cmds);
cmds.clear();
}
ReplyMarkups::const_iterator i = replyMarkups.constFind(msgId);
if (i == replyMarkups.cend() || true) return zeroMarkup;
return i.value();
}
void setProxySettings(QNetworkAccessManager &manager) { void setProxySettings(QNetworkAccessManager &manager) {
if (cConnectionType() == dbictHttpProxy) { if (cConnectionType() == dbictHttpProxy) {
const ConnectionProxy &p(cConnectionProxy()); const ConnectionProxy &p(cConnectionProxy());
@ -1946,9 +2089,9 @@ namespace App {
} }
} }
void sendBotCommand(const QString &cmd) { void sendBotCommand(const QString &cmd, MsgId replyTo) {
if (App::main()) { if (App::main()) {
App::main()->sendBotCommand(cmd); App::main()->sendBotCommand(cmd, replyTo);
} }
} }

View File

@ -35,6 +35,7 @@ typedef QHash<VideoData*, HistoryItemsMap> VideoItems;
typedef QHash<AudioData*, HistoryItemsMap> AudioItems; typedef QHash<AudioData*, HistoryItemsMap> AudioItems;
typedef QHash<DocumentData*, HistoryItemsMap> DocumentItems; typedef QHash<DocumentData*, HistoryItemsMap> DocumentItems;
typedef QHash<WebPageData*, HistoryItemsMap> WebPageItems; typedef QHash<WebPageData*, HistoryItemsMap> WebPageItems;
typedef QList<QList<QString> > ReplyMarkup;
enum RoundCorners { enum RoundCorners {
MaskCorners = 0x00, // for images MaskCorners = 0x00, // for images
@ -48,6 +49,9 @@ enum RoundCorners {
MediaviewSaveCorners, MediaviewSaveCorners,
EmojiHoverCorners, EmojiHoverCorners,
StickerHoverCorners, StickerHoverCorners,
BotKeyboardCorners,
BotKeyboardOverCorners,
BotKeyboardDownCorners,
InShadowCorners, // for photos without bg InShadowCorners, // for photos without bg
InSelectedShadowCorners, InSelectedShadowCorners,
@ -99,7 +103,7 @@ namespace App {
UserData *feedUsers(const MTPVector<MTPUser> &users); // returns last user UserData *feedUsers(const MTPVector<MTPUser> &users); // returns last user
ChatData *feedChats(const MTPVector<MTPChat> &chats); // returns last chat ChatData *feedChats(const MTPVector<MTPChat> &chats); // returns last chat
void feedParticipants(const MTPChatParticipants &p); void feedParticipants(const MTPChatParticipants &p, bool requestBotInfos = false);
void feedParticipantAdd(const MTPDupdateChatParticipantAdd &d); void feedParticipantAdd(const MTPDupdateChatParticipantAdd &d);
void feedParticipantDelete(const MTPDupdateChatParticipantDelete &d); void feedParticipantDelete(const MTPDupdateChatParticipantDelete &d);
void feedMsgs(const MTPVector<MTPMessage> &msgs, int msgsState = 0); // 2 - new read message, 1 - new unread message, 0 - not new message, -1 - searched message void feedMsgs(const MTPVector<MTPMessage> &msgs, int msgsState = 0); // 2 - new read message, 1 - new unread message, 0 - not new message, -1 - searched message
@ -222,10 +226,14 @@ namespace App {
void unregMuted(PeerData *peer); void unregMuted(PeerData *peer);
void updateMuted(); void updateMuted();
void feedReplyMarkup(MsgId msgId, const MTPReplyMarkup &markup);
void clearReplyMarkup(MsgId msgId);
const ReplyMarkup &replyMarkup(MsgId msgId);
void setProxySettings(QNetworkAccessManager &manager); void setProxySettings(QNetworkAccessManager &manager);
void setProxySettings(QTcpSocket &socket); void setProxySettings(QTcpSocket &socket);
void sendBotCommand(const QString &cmd); void sendBotCommand(const QString &cmd, MsgId replyTo = 0);
void searchByHashtag(const QString &tag); void searchByHashtag(const QString &tag);
void openUserByName(const QString &username, bool toProfile = false); void openUserByName(const QString &username, bool toProfile = false);
void joinGroupByHash(const QString &hash); void joinGroupByHash(const QString &hash);

View File

@ -640,10 +640,10 @@ void Application::checkMapVersion() {
psRegisterCustomScheme(); psRegisterCustomScheme();
if (Local::oldMapVersion()) { if (Local::oldMapVersion()) {
QString versionFeatures; QString versionFeatures;
if (DevChannel && Local::oldMapVersion() < 8023) { if (DevChannel && Local::oldMapVersion() < 8025) {
versionFeatures = QString::fromUtf8("\xe2\x80\x94 Improved sticker panel\n\xe2\x80\x94 Bug fixes and minor stuff");// .replace('@', qsl("@") + QChar(0x200D)); versionFeatures = QString::fromUtf8("\xe2\x80\x94 IPv6 connections support\n\xe2\x80\x94 Bug fixes and minor stuff");// .replace('@', qsl("@") + QChar(0x200D));
} else if (!DevChannel && Local::oldMapVersion() < 8024) { } else if (!DevChannel && Local::oldMapVersion() < 8024) {
versionFeatures = lang(lng_new_version_text).trimmed(); versionFeatures = lng_new_version_text(lt_blog_link, qsl("https://telegram.org/blog/bot-revolution"));// lang(lng_new_version_text).trimmed();
} }
if (!versionFeatures.isEmpty()) { if (!versionFeatures.isEmpty()) {
versionFeatures = lng_new_version_wrap(lt_version, QString::fromStdWString(AppVersionStr), lt_changes, versionFeatures, lt_link, qsl("https://desktop.telegram.org/#changelog")); versionFeatures = lng_new_version_wrap(lt_version, QString::fromStdWString(AppVersionStr), lt_changes, versionFeatures, lt_link, qsl("https://desktop.telegram.org/#changelog"));

Binary file not shown.

Before

Width:  |  Height:  |  Size: 166 KiB

After

Width:  |  Height:  |  Size: 167 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 218 KiB

After

Width:  |  Height:  |  Size: 219 KiB

View File

@ -23,7 +23,9 @@ Copyright (c) 2014 John Preston, https://desktop.telegram.org
#include "mainwidget.h" #include "mainwidget.h"
#include "window.h" #include "window.h"
ContactsInner::ContactsInner(bool creatingChat) : _chat(0), _creatingChat(creatingChat), #include "confirmbox.h"
ContactsInner::ContactsInner(bool creatingChat) : _chat(0), _bot(0), _creatingChat(creatingChat), _addToChat(0),
_contacts(&App::main()->contactsList()), _contacts(&App::main()->contactsList()),
_sel(0), _sel(0),
_filteredSel(-1), _filteredSel(-1),
@ -35,7 +37,7 @@ _addContactLnk(this, lang(lng_add_contact_button)) {
init(); init();
} }
ContactsInner::ContactsInner(ChatData *chat) : _chat(chat), _creatingChat(false), ContactsInner::ContactsInner(ChatData *chat) : _chat(chat), _bot(0), _creatingChat(false), _addToChat(0),
_contacts(&App::main()->contactsList()), _contacts(&App::main()->contactsList()),
_sel(0), _sel(0),
_filteredSel(-1), _filteredSel(-1),
@ -47,6 +49,24 @@ _addContactLnk(this, lang(lng_add_contact_button)) {
init(); init();
} }
ContactsInner::ContactsInner(UserData *bot) : _chat(0), _bot(bot), _creatingChat(false), _addToChat(0),
_contacts(new DialogsIndexed(DialogsSortByAdd)),
_sel(0),
_filteredSel(-1),
_mouseSel(false),
_selCount(0),
_searching(false),
_byUsernameSel(-1),
_addContactLnk(this, lang(lng_add_contact_button)) {
DialogsIndexed &v(App::main()->dialogsList());
for (DialogRow *r = v.list.begin; r != v.list.end; r = r->next) {
if (r->history->peer->chat && !r->history->peer->asChat()->forbidden) {
_contacts->addToEnd(r->history);
}
}
init();
}
void ContactsInner::init() { void ContactsInner::init() {
connect(&_addContactLnk, SIGNAL(clicked()), App::wnd(), SLOT(onShowAddContact())); connect(&_addContactLnk, SIGNAL(clicked()), App::wnd(), SLOT(onShowAddContact()));
@ -59,10 +79,21 @@ void ContactsInner::init() {
connect(App::main(), SIGNAL(dialogRowReplaced(DialogRow *, DialogRow *)), this, SLOT(onDialogRowReplaced(DialogRow *, DialogRow *))); connect(App::main(), SIGNAL(dialogRowReplaced(DialogRow *, DialogRow *)), this, SLOT(onDialogRowReplaced(DialogRow *, DialogRow *)));
connect(App::main(), SIGNAL(peerUpdated(PeerData*)), this, SLOT(peerUpdated(PeerData *))); connect(App::main(), SIGNAL(peerUpdated(PeerData*)), this, SLOT(peerUpdated(PeerData *)));
connect(App::main(), SIGNAL(peerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)), this, SLOT(peerUpdated(PeerData *))); connect(App::main(), SIGNAL(peerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)), this, SLOT(onPeerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)));
connect(App::main(), SIGNAL(peerPhotoChanged(PeerData *)), this, SLOT(peerUpdated(PeerData *))); connect(App::main(), SIGNAL(peerPhotoChanged(PeerData *)), this, SLOT(peerUpdated(PeerData *)));
} }
void ContactsInner::onPeerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars) {
if (bot()) {
_contacts->peerNameChanged(peer, oldNames, oldChars);
}
peerUpdated(peer);
}
void ContactsInner::onAddBot() {
App::main()->addParticipants(_addToChat, QVector<UserData*>(1, _bot));
}
void ContactsInner::peerUpdated(PeerData *peer) { void ContactsInner::peerUpdated(PeerData *peer) {
if (_chat && (!peer || peer == _chat)) { if (_chat && (!peer || peer == _chat)) {
if (_chat->forbidden) { if (_chat->forbidden) {
@ -81,8 +112,8 @@ void ContactsInner::peerUpdated(PeerData *peer) {
} }
} }
} }
} else if (!peer->chat) { } else {
ContactsData::iterator i = _contactsData.find(peer->asUser()); ContactsData::iterator i = _contactsData.find(peer);
if (i != _contactsData.cend()) { if (i != _contactsData.cend()) {
for (DialogRow *row = _contacts->list.begin; row->next; row = row->next) { for (DialogRow *row = _contacts->list.begin; row->next; row = row->next) {
if (row->attached == i.value()) row->attached = 0; if (row->attached == i.value()) row->attached = 0;
@ -136,14 +167,23 @@ void ContactsInner::loadProfilePhotos(int32 yFrom) {
ContactsInner::ContactData *ContactsInner::contactData(DialogRow *row) { ContactsInner::ContactData *ContactsInner::contactData(DialogRow *row) {
ContactData *data = (ContactData*)row->attached; ContactData *data = (ContactData*)row->attached;
if (!data) { if (!data) {
UserData *user = row->history->peer->asUser(); PeerData *peer = row->history->peer;
ContactsData::const_iterator i = _contactsData.constFind(user); ContactsData::const_iterator i = _contactsData.constFind(peer);
if (i == _contactsData.cend()) { if (i == _contactsData.cend()) {
_contactsData.insert(user, data = new ContactData()); _contactsData.insert(peer, data = new ContactData());
data->inchat = _chat ? _chat->participants.contains(user) : false; data->inchat = (_chat && !peer->chat) ? _chat->participants.contains(peer->asUser()) : false;
data->check = false; data->check = false;
data->name.setText(st::profileListNameFont, user->name, _textNameOptions); data->name.setText(st::profileListNameFont, peer->name, _textNameOptions);
data->online = App::onlineText(user, _time); if (peer->chat) {
ChatData *chat = peer->asChat();
if (chat->forbidden) {
data->online = lang(lng_chat_status_unaccessible);
} else {
data->online = lng_chat_status_members(lt_count, chat->count);
}
} else {
data->online = App::onlineText(peer->asUser(), _time);
}
} else { } else {
data = i.value(); data = i.value();
} }
@ -152,9 +192,11 @@ ContactsInner::ContactData *ContactsInner::contactData(DialogRow *row) {
return data; return data;
} }
void ContactsInner::paintDialog(QPainter &p, UserData *user, ContactData *data, bool sel) { void ContactsInner::paintDialog(QPainter &p, PeerData *peer, ContactData *data, bool sel) {
int32 left = st::profileListPadding.width(); int32 left = st::profileListPadding.width();
UserData *user = peer->chat ? 0 : peer->asUser();
if (data->inchat || data->check || _selCount + (_chat ? _chat->count : 0) >= cMaxGroupCount()) { if (data->inchat || data->check || _selCount + (_chat ? _chat->count : 0) >= cMaxGroupCount()) {
sel = false; sel = false;
} }
@ -163,7 +205,7 @@ void ContactsInner::paintDialog(QPainter &p, UserData *user, ContactData *data,
p.fillRect(0, 0, width(), 2 * st::profileListPadding.height() + st::profileListPhotoSize, ((data->inchat || data->check) ? st::profileActiveBG : st::profileHoverBG)->b); p.fillRect(0, 0, width(), 2 * st::profileListPadding.height() + st::profileListPhotoSize, ((data->inchat || data->check) ? st::profileActiveBG : st::profileHoverBG)->b);
} }
p.drawPixmap(left, st::profileListPadding.height(), user->photo->pix(st::profileListPhotoSize)); p.drawPixmap(left, st::profileListPadding.height(), peer->photo->pix(st::profileListPhotoSize));
if (data->inchat || data->check) { if (data->inchat || data->check) {
p.setPen(st::white->p); p.setPen(st::white->p);
@ -181,8 +223,7 @@ void ContactsInner::paintDialog(QPainter &p, UserData *user, ContactData *data,
p.drawPixmap(QPoint(width() - st::contactsImg.pxWidth() - st::profileCheckDeltaX, st::profileListPadding.height() + (st::profileListPhotoSize - st::contactsImg.pxHeight()) / 2 - st::profileCheckDeltaY), App::sprite(), st::contactsImg); p.drawPixmap(QPoint(width() - st::contactsImg.pxWidth() - st::profileCheckDeltaX, st::profileListPadding.height() + (st::profileListPhotoSize - st::contactsImg.pxHeight()) / 2 - st::profileCheckDeltaY), App::sprite(), st::contactsImg);
} }
bool uname = user && (data->online.at(0) == '@');
bool uname = (data->online.at(0) == '@');
p.setFont(st::profileSubFont->f); p.setFont(st::profileSubFont->f);
if (uname && !data->inchat && !data->check && !_lastQuery.isEmpty() && user->username.startsWith(_lastQuery, Qt::CaseInsensitive)) { if (uname && !data->inchat && !data->check && !_lastQuery.isEmpty() && user->username.startsWith(_lastQuery, Qt::CaseInsensitive)) {
int32 availw = width() - (left + st::profileListPhotoSize + st::profileListPadding.width() * 2); int32 availw = width() - (left + st::profileListPhotoSize + st::profileListPadding.width() * 2);
@ -200,8 +241,10 @@ void ContactsInner::paintDialog(QPainter &p, UserData *user, ContactData *data,
} else { } else {
if (data->inchat || data->check) { if (data->inchat || data->check) {
p.setPen(st::white->p); p.setPen(st::white->p);
} else if (user && (uname || App::onlineColorUse(user->onlineTill, _time))) {
p.setPen(st::profileOnlineColor->p);
} else { } else {
p.setPen(((uname || App::onlineColorUse(user->onlineTill, _time)) ? st::profileOnlineColor : st::profileOfflineColor)->p); p.setPen(st::profileOfflineColor->p);
} }
p.drawText(left + st::profileListPhotoSize + st::profileListPadding.width(), st::profileListPadding.height() + st::profileListPhotoSize - st::profileListStatusBottom, data->online); p.drawText(left + st::profileListPhotoSize + st::profileListPadding.width(), st::profileListPadding.height() + st::profileListPhotoSize - st::profileListStatusBottom, data->online);
} }
@ -224,7 +267,7 @@ void ContactsInner::paintEvent(QPaintEvent *e) {
DialogRow *drawFrom = _contacts->list.current; DialogRow *drawFrom = _contacts->list.current;
p.translate(0, drawFrom->pos * rh); p.translate(0, drawFrom->pos * rh);
while (drawFrom != _contacts->list.end && drawFrom->pos * rh < yTo) { while (drawFrom != _contacts->list.end && drawFrom->pos * rh < yTo) {
paintDialog(p, drawFrom->history->peer->asUser(), contactData(drawFrom), (drawFrom == _sel)); paintDialog(p, drawFrom->history->peer, contactData(drawFrom), (drawFrom == _sel));
p.translate(0, rh); p.translate(0, rh);
drawFrom = drawFrom->next; drawFrom = drawFrom->next;
} }
@ -253,13 +296,29 @@ void ContactsInner::paintEvent(QPaintEvent *e) {
} else { } else {
p.setFont(st::noContactsFont->f); p.setFont(st::noContactsFont->f);
p.setPen(st::noContactsColor->p); p.setPen(st::noContactsColor->p);
p.drawText(QRect(0, 0, width(), st::noContactsHeight - ((cContactsReceived() && !_searching) ? st::noContactsFont->height : 0)), lang((cContactsReceived() && !_searching) ? lng_no_contacts : lng_contacts_loading), style::al_center); QString text;
int32 skip = 0;
if (bot()) {
text = lang(cDialogsReceived() ? lng_bot_no_groups : lng_contacts_loading);
} else if (cContactsReceived() && !_searching) {
text = lang(lng_no_contacts);
skip = st::noContactsFont->height;
} else {
text = lang(lng_contacts_loading);
}
p.drawText(QRect(0, 0, width(), st::noContactsHeight - skip), text, style::al_center);
} }
} else { } else {
if (_filtered.isEmpty() && _byUsernameFiltered.isEmpty()) { if (_filtered.isEmpty() && _byUsernameFiltered.isEmpty()) {
p.setFont(st::noContactsFont->f); p.setFont(st::noContactsFont->f);
p.setPen(st::noContactsColor->p); p.setPen(st::noContactsColor->p);
p.drawText(QRect(0, 0, width(), st::noContactsHeight), lang((cContactsReceived() && !_searching) ? lng_contacts_not_found : lng_contacts_loading), style::al_center); QString text;
if (bot()) {
text = lang(cDialogsReceived() ? lng_bot_groups_not_found : lng_contacts_loading);
} else {
text = lang((cContactsReceived() && !_searching) ? lng_contacts_not_found : lng_contacts_loading);
}
p.drawText(QRect(0, 0, width(), st::noContactsHeight), text, style::al_center);
} else { } else {
if (!_filtered.isEmpty()) { if (!_filtered.isEmpty()) {
int32 from = (yFrom >= 0) ? (yFrom / rh) : 0; int32 from = (yFrom >= 0) ? (yFrom / rh) : 0;
@ -269,7 +328,7 @@ void ContactsInner::paintEvent(QPaintEvent *e) {
p.translate(0, from * rh); p.translate(0, from * rh);
for (; from < to; ++from) { for (; from < to; ++from) {
paintDialog(p, _filtered[from]->history->peer->asUser(), contactData(_filtered[from]), (_filteredSel == from)); paintDialog(p, _filtered[from]->history->peer, contactData(_filtered[from]), (_filteredSel == from));
p.translate(0, rh); p.translate(0, rh);
} }
} }
@ -371,25 +430,32 @@ void ContactsInner::chooseParticipant() {
} }
} else { } else {
int32 rh = st::profileListPhotoSize + st::profileListPadding.height() * 2, from; int32 rh = st::profileListPhotoSize + st::profileListPadding.height() * 2, from;
PeerId peer = 0; PeerData *peer = 0;
if (_filter.isEmpty()) { if (_filter.isEmpty()) {
if (_byUsernameSel >= 0 && _byUsernameSel < _byUsername.size()) { if (_byUsernameSel >= 0 && _byUsernameSel < _byUsername.size()) {
peer = _byUsername[_byUsernameSel]->id; peer = _byUsername[_byUsernameSel];
} else { } else if (_sel) {
peer = _sel ? _sel->history->peer->id : 0; peer = _sel->history->peer;
} }
} else { } else {
if (_byUsernameSel >= 0 && _byUsernameSel < _byUsernameFiltered.size()) { if (_byUsernameSel >= 0 && _byUsernameSel < _byUsernameFiltered.size()) {
peer = _byUsernameFiltered[_byUsernameSel]->id; peer = _byUsernameFiltered[_byUsernameSel];
} else { } else {
if (_filteredSel < 0 || _filteredSel >= _filtered.size()) return; if (_filteredSel < 0 || _filteredSel >= _filtered.size()) return;
peer = _filtered[_filteredSel]->history->peer->id; peer = _filtered[_filteredSel]->history->peer;
} }
} }
if (peer) { if (peer) {
App::wnd()->hideSettings(true); if (bot() && peer->chat) {
App::main()->showPeer(peer, 0, false, true); _addToChat = peer->asChat();
App::wnd()->hideLayer(); ConfirmBox *box = new ConfirmBox(lng_bot_sure_invite(lt_group, peer->name));
connect(box, SIGNAL(confirmed()), this, SLOT(onAddBot()));
App::wnd()->replaceLayer(box);
} else {
App::wnd()->hideSettings(true);
App::main()->showPeer(peer->id, 0, false, true);
App::wnd()->hideLayer();
}
} }
} }
parentWidget()->update(); parentWidget()->update();
@ -562,8 +628,10 @@ void ContactsInner::updateFilter(QString filter) {
refresh(); refresh();
_searching = true; if (!bot()) {
emit searchByUsername(); _searching = true;
emit searchByUsername();
}
} }
if (parentWidget()) parentWidget()->update(); if (parentWidget()) parentWidget()->update();
loadProfilePhotos(0); loadProfilePhotos(0);
@ -612,6 +680,8 @@ void ContactsInner::peopleReceived(const QString &query, const QVector<MTPContac
} }
if (j == already) { if (j == already) {
UserData *u = App::user(uid); UserData *u = App::user(uid);
if (u->botInfo && u->botInfo->cantJoinGroups && (_chat || _creatingChat)) continue; // skip bot's that can't be invited to groups
ContactData *d = new ContactData(); ContactData *d = new ContactData();
_byUsernameDatas.push_back(d); _byUsernameDatas.push_back(d);
d->inchat = _chat ? _chat->participants.contains(u) : false; d->inchat = _chat ? _chat->participants.contains(u) : false;
@ -634,7 +704,7 @@ void ContactsInner::refresh() {
if (!_addContactLnk.isHidden()) _addContactLnk.hide(); if (!_addContactLnk.isHidden()) _addContactLnk.hide();
resize(width(), (_contacts->list.count * rh) + (_byUsername.isEmpty() ? 0 : (st::searchedBarHeight + _byUsername.size() * rh))); resize(width(), (_contacts->list.count * rh) + (_byUsername.isEmpty() ? 0 : (st::searchedBarHeight + _byUsername.size() * rh)));
} else { } else {
if (cContactsReceived()) { if (cContactsReceived() && !bot()) {
if (_addContactLnk.isHidden()) _addContactLnk.show(); if (_addContactLnk.isHidden()) _addContactLnk.show();
} else { } else {
if (!_addContactLnk.isHidden()) _addContactLnk.hide(); if (!_addContactLnk.isHidden()) _addContactLnk.hide();
@ -656,6 +726,10 @@ ChatData *ContactsInner::chat() const {
return _chat; return _chat;
} }
UserData *ContactsInner::bot() const {
return _bot;
}
bool ContactsInner::creatingChat() const { bool ContactsInner::creatingChat() const {
return _creatingChat; return _creatingChat;
} }
@ -664,6 +738,7 @@ ContactsInner::~ContactsInner() {
for (ContactsData::iterator i = _contactsData.begin(), e = _contactsData.end(); i != e; ++i) { for (ContactsData::iterator i = _contactsData.begin(), e = _contactsData.end(); i != e; ++i) {
delete *i; delete *i;
} }
if (_bot) delete _contacts;
} }
void ContactsInner::resizeEvent(QResizeEvent *e) { void ContactsInner::resizeEvent(QResizeEvent *e) {
@ -796,8 +871,8 @@ QVector<UserData*> ContactsInner::selected() {
QVector<UserData*> result; QVector<UserData*> result;
result.reserve(_contactsData.size()); result.reserve(_contactsData.size());
for (ContactsData::const_iterator i = _contactsData.cbegin(), e = _contactsData.cend(); i != e; ++i) { for (ContactsData::const_iterator i = _contactsData.cbegin(), e = _contactsData.cend(); i != e; ++i) {
if (i.value()->check) { if (i.value()->check && !i.key()->chat) {
result.push_back(i.key()); result.push_back(i.key()->asUser());
} }
} }
for (int32 i = 0, l = _byUsername.size(); i < l; ++i) { for (int32 i = 0, l = _byUsername.size(); i < l; ++i) {
@ -812,7 +887,7 @@ QVector<MTPInputUser> ContactsInner::selectedInputs() {
QVector<MTPInputUser> result; QVector<MTPInputUser> result;
result.reserve(_contactsData.size()); result.reserve(_contactsData.size());
for (ContactsData::const_iterator i = _contactsData.cbegin(), e = _contactsData.cend(); i != e; ++i) { for (ContactsData::const_iterator i = _contactsData.cbegin(), e = _contactsData.cend(); i != e; ++i) {
if (i.value()->check) { if (i.value()->check && !i.key()->chat) {
result.push_back(i.key()->inputUser); result.push_back(i.key()->inputUser);
} }
} }
@ -854,6 +929,14 @@ _cancel(this, lang(lng_cancel), st::btnSelectCancel) {
init(); init();
} }
ContactsBox::ContactsBox(UserData *bot) : ItemListBox(st::boxNoTopScroll), _inner(bot),
_addContact(this, lang(lng_add_contact_button), st::contactsAdd),
_filter(this, st::contactsFilter, lang(lng_participant_filter)),
_next(this, lang(lng_create_group_next), st::btnSelectDone),
_cancel(this, lang(lng_cancel), st::contactsClose) {
init();
}
void ContactsBox::init() { void ContactsBox::init() {
ItemListBox::init(&_inner, _cancel.height(), st::contactsAdd.height + st::newGroupNamePadding.top() + _filter.height() + st::newGroupNamePadding.bottom()); ItemListBox::init(&_inner, _cancel.height(), st::contactsAdd.height + st::newGroupNamePadding.top() + _filter.height() + st::newGroupNamePadding.bottom());
@ -961,14 +1044,17 @@ void ContactsBox::hideAll() {
void ContactsBox::showAll() { void ContactsBox::showAll() {
ItemListBox::showAll(); ItemListBox::showAll();
_addContact.show();
_filter.show(); _filter.show();
if (_inner.chat() || _inner.creatingChat()) { if (_inner.chat() || _inner.creatingChat()) {
_next.show(); _next.show();
_addContact.hide(); _addContact.hide();
} else { } else {
_next.hide(); _next.hide();
_addContact.show(); if (_inner.bot()) {
_addContact.hide();
} else {
_addContact.show();
}
} }
_cancel.show(); _cancel.show();
} }
@ -1010,6 +1096,8 @@ void ContactsBox::paintEvent(QPaintEvent *e) {
// paint button sep // paint button sep
p.fillRect(st::btnSelectCancel.width, size().height() - st::btnSelectCancel.height, st::lineWidth, st::btnSelectCancel.height, st::btnSelectSep->b); p.fillRect(st::btnSelectCancel.width, size().height() - st::btnSelectCancel.height, st::lineWidth, st::btnSelectCancel.height, st::btnSelectSep->b);
} else if (_inner.bot()) {
paintTitle(p, lang(lng_bot_choose_group), true);
} else { } else {
paintTitle(p, lang(lng_contacts_header), true); paintTitle(p, lang(lng_contacts_header), true);
} }

View File

@ -30,6 +30,7 @@ public:
ContactsInner(bool creatingChat); ContactsInner(bool creatingChat);
ContactsInner(ChatData *chat); ContactsInner(ChatData *chat);
ContactsInner(UserData *bot);
void init(); void init();
void paintEvent(QPaintEvent *e); void paintEvent(QPaintEvent *e);
@ -39,7 +40,7 @@ public:
void mousePressEvent(QMouseEvent *e); void mousePressEvent(QMouseEvent *e);
void resizeEvent(QResizeEvent *e); void resizeEvent(QResizeEvent *e);
void paintDialog(QPainter &p, UserData *user, ContactData *data, bool sel); void paintDialog(QPainter &p, PeerData *peer, ContactData *data, bool sel);
void updateFilter(QString filter = QString()); void updateFilter(QString filter = QString());
void selectSkip(int32 dir); void selectSkip(int32 dir);
@ -59,6 +60,7 @@ public:
void refresh(); void refresh();
ChatData *chat() const; ChatData *chat() const;
UserData *bot() const;
bool creatingChat() const; bool creatingChat() const;
~ContactsInner(); ~ContactsInner();
@ -75,11 +77,17 @@ public slots:
void updateSel(); void updateSel();
void peerUpdated(PeerData *peer); void peerUpdated(PeerData *peer);
void onPeerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars);
void onAddBot();
private: private:
ChatData *_chat; ChatData *_chat;
UserData *_bot;
bool _creatingChat; bool _creatingChat;
ChatData *_addToChat;
int32 _time; int32 _time;
@ -99,7 +107,7 @@ private:
bool inchat; bool inchat;
bool check; bool check;
}; };
typedef QMap<UserData*, ContactData*> ContactsData; typedef QMap<PeerData*, ContactData*> ContactsData;
ContactsData _contactsData; ContactsData _contactsData;
ContactData *contactData(DialogRow *row); ContactData *contactData(DialogRow *row);
@ -125,6 +133,7 @@ public:
ContactsBox(bool creatingChat = false); ContactsBox(bool creatingChat = false);
ContactsBox(ChatData *chat); ContactsBox(ChatData *chat);
ContactsBox(UserData *bot);
void keyPressEvent(QKeyEvent *e); void keyPressEvent(QKeyEvent *e);
void paintEvent(QPaintEvent *e); void paintEvent(QPaintEvent *e);
void resizeEvent(QResizeEvent *e); void resizeEvent(QResizeEvent *e);

View File

@ -17,9 +17,9 @@ Copyright (c) 2014 John Preston, https://desktop.telegram.org
*/ */
#pragma once #pragma once
static const int32 AppVersion = 8024; static const int32 AppVersion = 8025;
static const wchar_t *AppVersionStr = L"0.8.24"; static const wchar_t *AppVersionStr = L"0.8.25";
static const bool DevChannel = false; static const bool DevChannel = true;
static const wchar_t *AppNameOld = L"Telegram Win (Unofficial)"; static const wchar_t *AppNameOld = L"Telegram Win (Unofficial)";
static const wchar_t *AppName = L"Telegram Desktop"; static const wchar_t *AppName = L"Telegram Desktop";

View File

@ -28,9 +28,9 @@ Copyright (c) 2014 John Preston, https://desktop.telegram.org
#include "localstorage.h" #include "localstorage.h"
DialogsListWidget::DialogsListWidget(QWidget *parent, MainWidget *main) : QWidget(parent), DialogsListWidget::DialogsListWidget(QWidget *parent, MainWidget *main) : QWidget(parent),
dialogs(false), dialogs(DialogsSortByDate),
contactsNoDialogs(true), contactsNoDialogs(DialogsSortByName),
contacts(true), contacts(DialogsSortByName),
sel(0), sel(0),
contactSel(false), contactSel(false),
selByMouse(false), selByMouse(false),
@ -64,6 +64,8 @@ int32 DialogsListWidget::searchedOffset() const {
} }
void DialogsListWidget::paintEvent(QPaintEvent *e) { void DialogsListWidget::paintEvent(QPaintEvent *e) {
if (!App::main()) return;
QRect r(e->rect()); QRect r(e->rect());
bool trivial = (rect() == r); bool trivial = (rect() == r);
@ -95,7 +97,7 @@ void DialogsListWidget::paintEvent(QPaintEvent *e) {
} }
p.translate(0, from * st::mentionHeight); p.translate(0, from * st::mentionHeight);
if (from < hashtagResults.size()) { if (from < hashtagResults.size()) {
int32 to = (r.bottom() / int32(st::mentionHeight)) + 1, w = width(); int32 to = (r.bottom() / int32(st::mentionHeight)) + 1, w = width(), htagwidth = w - st::dlgPaddingHor * 2;
if (to > hashtagResults.size()) to = hashtagResults.size(); if (to > hashtagResults.size()) to = hashtagResults.size();
p.setFont(st::mentionFont->f); p.setFont(st::mentionFont->f);
p.setPen(st::black->p); p.setPen(st::black->p);
@ -106,8 +108,27 @@ void DialogsListWidget::paintEvent(QPaintEvent *e) {
int skip = (st::mentionHeight - st::notifyClose.icon.pxHeight()) / 2; int skip = (st::mentionHeight - st::notifyClose.icon.pxHeight()) / 2;
p.drawPixmap(QPoint(w - st::notifyClose.icon.pxWidth() - skip, skip), App::sprite(), st::notifyClose.icon); p.drawPixmap(QPoint(w - st::notifyClose.icon.pxWidth() - skip, skip), App::sprite(), st::notifyClose.icon);
} }
QString tag = st::mentionFont->m.elidedText('#' + hashtagResults.at(from), Qt::ElideRight, w - st::dlgPaddingHor * 2);
p.drawText(st::dlgPaddingHor, st::mentionTop + st::mentionFont->ascent, tag); QString first = (_hashtagFilter.size() < 2) ? QString() : ('#' + hashtagResults.at(from).mid(0, _hashtagFilter.size() - 1)), second = (_hashtagFilter.size() < 2) ? ('#' + hashtagResults.at(from)) : hashtagResults.at(from).mid(_hashtagFilter.size() - 1);
int32 firstwidth = st::mentionFont->m.width(first), secondwidth = st::mentionFont->m.width(second);
if (htagwidth < firstwidth + secondwidth) {
if (htagwidth < firstwidth + st::mentionFont->elidew) {
first = st::mentionFont->m.elidedText(first + second, Qt::ElideRight, htagwidth);
second = QString();
} else {
second = st::mentionFont->m.elidedText(second, Qt::ElideRight, htagwidth - firstwidth);
}
}
p.setFont(st::mentionFont->f);
if (!first.isEmpty()) {
p.setPen(st::profileOnlineColor->p);
p.drawText(st::dlgPaddingHor, st::mentionTop + st::mentionFont->ascent, first);
}
if (!second.isEmpty()) {
p.setPen(st::profileOfflineColor->p);
p.drawText(st::dlgPaddingHor + firstwidth, st::mentionTop + st::mentionFont->ascent, second);
}
p.translate(0, st::mentionHeight); p.translate(0, st::mentionHeight);
} }
} }
@ -607,8 +628,9 @@ void DialogsListWidget::onFilterUpdate(QString newFilter, bool force) {
} }
} }
void DialogsListWidget::onHashtagFilterUpdate(QString newFilter) { void DialogsListWidget::onHashtagFilterUpdate(QStringRef newFilter) {
if (newFilter.isEmpty() || newFilter.at(0) != '#') { if (newFilter.isEmpty() || newFilter.at(0) != '#') {
_hashtagFilter = QString();
if (!hashtagResults.isEmpty()) { if (!hashtagResults.isEmpty()) {
hashtagResults.clear(); hashtagResults.clear();
refresh(true); refresh(true);
@ -616,6 +638,7 @@ void DialogsListWidget::onHashtagFilterUpdate(QString newFilter) {
} }
return; return;
} }
_hashtagFilter = newFilter.toString();
if (cRecentSearchHashtags().isEmpty() && cRecentWriteHashtags().isEmpty()) { if (cRecentSearchHashtags().isEmpty() && cRecentWriteHashtags().isEmpty()) {
Local::readRecentHashtags(); Local::readRecentHashtags();
} }
@ -624,7 +647,7 @@ void DialogsListWidget::onHashtagFilterUpdate(QString newFilter) {
if (!recent.isEmpty()) { if (!recent.isEmpty()) {
hashtagResults.reserve(qMin(recent.size(), 5)); hashtagResults.reserve(qMin(recent.size(), 5));
for (RecentHashtagPack::const_iterator i = recent.cbegin(), e = recent.cend(); i != e; ++i) { for (RecentHashtagPack::const_iterator i = recent.cbegin(), e = recent.cend(); i != e; ++i) {
if (i->first.startsWith(newFilter.midRef(1), Qt::CaseInsensitive) && i->first.size() + 1 != newFilter.size()) { if (i->first.startsWith(_hashtagFilter.midRef(1), Qt::CaseInsensitive) && i->first.size() + 1 != newFilter.size()) {
hashtagResults.push_back(i->first); hashtagResults.push_back(i->first);
if (hashtagResults.size() == 5) break; if (hashtagResults.size() == 5) break;
} }
@ -1604,7 +1627,10 @@ void DialogsWidget::onSearchMore(MsgId minMsgId) {
void DialogsWidget::loadDialogs() { void DialogsWidget::loadDialogs() {
if (dlgPreloading) return; if (dlgPreloading) return;
if (dlgCount >= 0 && dlgOffset >= dlgCount) return; if (dlgCount >= 0 && dlgOffset >= dlgCount) {
cSetDialogsReceived(true);
return;
}
int32 loadCount = dlgOffset ? DialogsPerPage : DialogsFirstLoad; int32 loadCount = dlgOffset ? DialogsPerPage : DialogsFirstLoad;
dlgPreloading = MTP::send(MTPmessages_GetDialogs(MTP_int(dlgOffset), MTP_int(0), MTP_int(loadCount)), rpcDone(&DialogsWidget::dialogsReceived), rpcFail(&DialogsWidget::dialogsFailed)); dlgPreloading = MTP::send(MTPmessages_GetDialogs(MTP_int(dlgOffset), MTP_int(0), MTP_int(loadCount)), rpcDone(&DialogsWidget::dialogsReceived), rpcFail(&DialogsWidget::dialogsFailed));
@ -1752,12 +1778,13 @@ void DialogsWidget::onFilterUpdate(bool force) {
void DialogsWidget::onFilterCursorMoved(int from, int to) { void DialogsWidget::onFilterCursorMoved(int from, int to) {
if (to < 0) to = _filter.cursorPosition(); if (to < 0) to = _filter.cursorPosition();
QString t = _filter.text(), r; QString t = _filter.text();
QStringRef r;
for (int start = to; start > 0;) { for (int start = to; start > 0;) {
--start; --start;
if (t.size() <= start) break; if (t.size() <= start) break;
if (t.at(start) == '#') { if (t.at(start) == '#') {
r = t.mid(start, to - start); r = t.midRef(start, to - start);
break; break;
} }
if (!t.at(start).isLetterOrNumber() && t.at(start) != '_') break; if (!t.at(start).isLetterOrNumber() && t.at(start) != '_') break;
@ -1886,6 +1913,10 @@ DialogsIndexed &DialogsWidget::contactsList() {
return list.contactsList(); return list.contactsList();
} }
DialogsIndexed &DialogsWidget::dialogsList() {
return list.dialogsList();
}
void DialogsWidget::onAddContact() { void DialogsWidget::onAddContact() {
App::wnd()->replaceLayer(new AddContactBox()); App::wnd()->replaceLayer(new AddContactBox());
} }

View File

@ -94,7 +94,7 @@ public:
bool hasFilteredResults() const; bool hasFilteredResults() const;
void onFilterUpdate(QString newFilter, bool force = false); void onFilterUpdate(QString newFilter, bool force = false);
void onHashtagFilterUpdate(QString newFilter); void onHashtagFilterUpdate(QStringRef newFilter);
void itemRemoved(HistoryItem *item); void itemRemoved(HistoryItem *item);
void itemReplaced(HistoryItem *oldItem, HistoryItem *newItem); void itemReplaced(HistoryItem *oldItem, HistoryItem *newItem);
@ -130,7 +130,7 @@ private:
bool contactSel; bool contactSel;
bool selByMouse; bool selByMouse;
QString filter; QString filter, _hashtagFilter;
QStringList hashtagResults; QStringList hashtagResults;
int32 hashtagSel; int32 hashtagSel;
@ -197,6 +197,7 @@ public:
void removeContact(UserData *user); void removeContact(UserData *user);
DialogsIndexed &contactsList(); DialogsIndexed &contactsList();
DialogsIndexed &dialogsList();
void enableShadow(bool enable = true); void enableShadow(bool enable = true);

View File

@ -779,7 +779,7 @@ void EmojiPanInner::paintEvent(QPaintEvent *e) {
int32 index = i * EmojiPanPerRow + j; int32 index = i * EmojiPanPerRow + j;
if (index >= size) break; if (index >= size) break;
float64 hover = (!_picker.isHidden() && c * emojiTabShift + index == _pickerSel) ? 1 : _hovers[c][index]; float64 hover = (!_picker.isHidden() && c * MatrixRowShift + index == _pickerSel) ? 1 : _hovers[c][index];
QPoint w(st::emojiPanPadding + j * st::emojiPanSize.width(), y + i * st::emojiPanSize.height()); QPoint w(st::emojiPanPadding + j * st::emojiPanSize.width(), y + i * st::emojiPanSize.height());
if (hover > 0) { if (hover > 0) {
@ -817,7 +817,7 @@ void EmojiPanInner::mousePressEvent(QMouseEvent *e) {
_pressedSel = _selected; _pressedSel = _selected;
if (_selected >= 0 && _selected != SwitcherSelected) { if (_selected >= 0 && _selected != SwitcherSelected) {
int tab = (_selected / emojiTabShift), sel = _selected % emojiTabShift; int tab = (_selected / MatrixRowShift), sel = _selected % MatrixRowShift;
if (tab < emojiTabCount && sel < _emojis[tab].size() && _emojis[tab][sel]->color) { if (tab < emojiTabCount && sel < _emojis[tab].size() && _emojis[tab][sel]->color) {
_pickerSel = _selected; _pickerSel = _selected;
if (cEmojiVariants().constFind(_emojis[tab][sel]->code) == cEmojiVariants().cend()) { if (cEmojiVariants().constFind(_emojis[tab][sel]->code) == cEmojiVariants().cend()) {
@ -838,7 +838,7 @@ void EmojiPanInner::mouseReleaseEvent(QMouseEvent *e) {
if (_picker.rect().contains(_picker.mapFromGlobal(_lastMousePos))) { if (_picker.rect().contains(_picker.mapFromGlobal(_lastMousePos))) {
return _picker.mouseReleaseEvent(0); return _picker.mouseReleaseEvent(0);
} else if (_pickerSel >= 0) { } else if (_pickerSel >= 0) {
int tab = (_pickerSel / emojiTabShift), sel = _pickerSel % emojiTabShift; int tab = (_pickerSel / MatrixRowShift), sel = _pickerSel % MatrixRowShift;
if (tab < emojiTabCount && sel < _emojis[tab].size() && _emojis[tab][sel]->color) { if (tab < emojiTabCount && sel < _emojis[tab].size() && _emojis[tab][sel]->color) {
if (cEmojiVariants().constFind(_emojis[tab][sel]->code) != cEmojiVariants().cend()) { if (cEmojiVariants().constFind(_emojis[tab][sel]->code) != cEmojiVariants().cend()) {
_picker.hideStart(); _picker.hideStart();
@ -860,11 +860,11 @@ void EmojiPanInner::mouseReleaseEvent(QMouseEvent *e) {
return; return;
} }
if (_selected >= emojiTabCount * emojiTabShift) { if (_selected >= emojiTabCount * MatrixRowShift) {
return; return;
} }
int tab = (_selected / emojiTabShift), sel = _selected % emojiTabShift; int tab = (_selected / MatrixRowShift), sel = _selected % MatrixRowShift;
if (sel < _emojis[tab].size()) { if (sel < _emojis[tab].size()) {
EmojiPtr emoji(_emojis[tab][sel]); EmojiPtr emoji(_emojis[tab][sel]);
if (emoji->color && !_picker.isHidden()) return; if (emoji->color && !_picker.isHidden()) return;
@ -917,7 +917,7 @@ void EmojiPanInner::onSaveConfig() {
} }
void EmojiPanInner::onShowPicker() { void EmojiPanInner::onShowPicker() {
int tab = (_pickerSel / emojiTabShift), sel = _pickerSel % emojiTabShift; int tab = (_pickerSel / MatrixRowShift), sel = _pickerSel % MatrixRowShift;
if (tab < emojiTabCount && sel < _emojis[tab].size() && _emojis[tab][sel]->color) { if (tab < emojiTabCount && sel < _emojis[tab].size() && _emojis[tab][sel]->color) {
int32 y = 0; int32 y = 0;
for (int c = 0; c <= tab; ++c) { for (int c = 0; c <= tab; ++c) {
@ -952,7 +952,7 @@ void EmojiPanInner::onColorSelected(EmojiPtr emoji) {
cRefEmojiVariants().insert(emoji->code, emojiKey(emoji)); cRefEmojiVariants().insert(emoji->code, emojiKey(emoji));
} }
if (_pickerSel >= 0) { if (_pickerSel >= 0) {
int tab = (_pickerSel / emojiTabShift), sel = _pickerSel % emojiTabShift; int tab = (_pickerSel / MatrixRowShift), sel = _pickerSel % MatrixRowShift;
if (tab >= 0 && tab < emojiTabCount) { if (tab >= 0 && tab < emojiTabCount) {
_emojis[tab][sel] = emoji; _emojis[tab][sel] = emoji;
update(); update();
@ -991,7 +991,7 @@ void EmojiPanInner::clearSelection(bool fast) {
_lastMousePos = mapToGlobal(QPoint(-10, -10)); _lastMousePos = mapToGlobal(QPoint(-10, -10));
if (fast) { if (fast) {
for (Animations::const_iterator i = _animations.cbegin(); i != _animations.cend(); ++i) { for (Animations::const_iterator i = _animations.cbegin(); i != _animations.cend(); ++i) {
int index = qAbs(i.key()) - 1, tab = (index / emojiTabShift), sel = index % emojiTabShift; int index = qAbs(i.key()) - 1, tab = (index / MatrixRowShift), sel = index % MatrixRowShift;
(index == SwitcherSelected ? _switcherHover : _hovers[tab][sel]) = 0; (index == SwitcherSelected ? _switcherHover : _hovers[tab][sel]) = 0;
} }
_animations.clear(); _animations.clear();
@ -1055,7 +1055,7 @@ void EmojiPanInner::updateSelected() {
if (selIndex >= _emojis[c].size()) { if (selIndex >= _emojis[c].size()) {
selIndex = -1; selIndex = -1;
} else { } else {
selIndex += c * emojiTabShift; selIndex += c * MatrixRowShift;
} }
} }
break; break;
@ -1098,7 +1098,7 @@ void EmojiPanInner::updateSelected() {
bool EmojiPanInner::animStep(float64 ms) { bool EmojiPanInner::animStep(float64 ms) {
uint64 now = getms(); uint64 now = getms();
for (Animations::iterator i = _animations.begin(); i != _animations.end();) { for (Animations::iterator i = _animations.begin(); i != _animations.end();) {
int index = qAbs(i.key()) - 1, tab = (index / emojiTabShift), sel = index % emojiTabShift; int index = qAbs(i.key()) - 1, tab = (index / MatrixRowShift), sel = index % MatrixRowShift;
float64 dt = float64(now - i.value()) / st::emojiPanDuration; float64 dt = float64(now - i.value()) / st::emojiPanDuration;
if (dt >= 1) { if (dt >= 1) {
(index == SwitcherSelected ? _switcherHover : _hovers[tab][sel]) = (i.key() > 0) ? 1 : 0; (index == SwitcherSelected ? _switcherHover : _hovers[tab][sel]) = (i.key() > 0) ? 1 : 0;
@ -1293,11 +1293,11 @@ void StickerPanInner::mouseReleaseEvent(QMouseEvent *e) {
emit switchToEmoji(); emit switchToEmoji();
return; return;
} }
if (_selected >= emojiTabShift * _setIds.size()) { if (_selected >= MatrixRowShift * _setIds.size()) {
return; return;
} }
int tab = (_selected / emojiTabShift), sel = _selected % emojiTabShift; int tab = (_selected / MatrixRowShift), sel = _selected % MatrixRowShift;
if (_setIds[tab] == RecentStickerSetId && sel >= _sets[tab].size() && sel < _sets[tab].size() * 2 && _custom.at(sel - _sets[tab].size())) { if (_setIds[tab] == RecentStickerSetId && sel >= _sets[tab].size() && sel < _sets[tab].size() * 2 && _custom.at(sel - _sets[tab].size())) {
clearSelection(true); clearSelection(true);
bool refresh = false; bool refresh = false;
@ -1362,7 +1362,7 @@ void StickerPanInner::clearSelection(bool fast) {
_lastMousePos = mapToGlobal(QPoint(-10, -10)); _lastMousePos = mapToGlobal(QPoint(-10, -10));
if (fast) { if (fast) {
for (Animations::const_iterator i = _animations.cbegin(); i != _animations.cend(); ++i) { for (Animations::const_iterator i = _animations.cbegin(); i != _animations.cend(); ++i) {
int index = qAbs(i.key()) - 1, tab = (index / emojiTabShift), sel = index % emojiTabShift; int index = qAbs(i.key()) - 1, tab = (index / MatrixRowShift), sel = index % MatrixRowShift;
(index == SwitcherSelected ? _switcherHover : _hovers[tab][sel]) = 0; (index == SwitcherSelected ? _switcherHover : _hovers[tab][sel]) = 0;
} }
_animations.clear(); _animations.clear();
@ -1554,7 +1554,7 @@ void StickerPanInner::updateSelected() {
ytill = y + st::emojiPanHeader + ((cnt / StickerPanPerRow) + ((cnt % StickerPanPerRow) ? 1 : 0)) * st::stickerPanSize.height(); ytill = y + st::emojiPanHeader + ((cnt / StickerPanPerRow) + ((cnt % StickerPanPerRow) ? 1 : 0)) * st::stickerPanSize.height();
if (p.y() >= y && p.y() < ytill) { if (p.y() >= y && p.y() < ytill) {
if (!special && p.y() >= y && p.y() < y + st::emojiPanHeader && sx + st::stickerPanPadding >= width() - st::emojiPanHeaderLeft - st::notifyClose.icon.pxWidth() && sx + st::stickerPanPadding < width() - st::emojiPanHeaderLeft) { if (!special && p.y() >= y && p.y() < y + st::emojiPanHeader && sx + st::stickerPanPadding >= width() - st::emojiPanHeaderLeft - st::notifyClose.icon.pxWidth() && sx + st::stickerPanPadding < width() - st::emojiPanHeaderLeft) {
selIndex = c * emojiTabShift + _sets[c].size(); selIndex = c * MatrixRowShift + _sets[c].size();
} else { } else {
y += st::emojiPanHeader; y += st::emojiPanHeader;
if (p.y() >= y && sx >= 0 && sx < StickerPanPerRow * st::stickerPanSize.width()) { if (p.y() >= y && sx >= 0 && sx < StickerPanPerRow * st::stickerPanSize.width()) {
@ -1568,7 +1568,7 @@ void StickerPanInner::updateSelected() {
selIndex += _sets[c].size(); selIndex += _sets[c].size();
} }
} }
selIndex += c * emojiTabShift; selIndex += c * MatrixRowShift;
} }
} }
} }
@ -1578,12 +1578,12 @@ void StickerPanInner::updateSelected() {
} }
bool startanim = false; bool startanim = false;
int oldSel = _selected, oldSelTab = oldSel / emojiTabShift, xOldSel = -1, newSel = selIndex, newSelTab = newSel / emojiTabShift, xNewSel = -1; int oldSel = _selected, oldSelTab = oldSel / MatrixRowShift, xOldSel = -1, newSel = selIndex, newSelTab = newSel / MatrixRowShift, xNewSel = -1;
if (oldSel >= 0 && oldSelTab < _setIds.size() && _setIds[oldSelTab] == RecentStickerSetId && oldSel >= oldSelTab * emojiTabShift + _sets[oldSelTab].size()) { if (oldSel >= 0 && oldSelTab < _setIds.size() && _setIds[oldSelTab] == RecentStickerSetId && oldSel >= oldSelTab * MatrixRowShift + _sets[oldSelTab].size()) {
xOldSel = oldSel; xOldSel = oldSel;
oldSel -= _sets[oldSelTab].size(); oldSel -= _sets[oldSelTab].size();
} }
if (newSel >= 0 && newSelTab < _setIds.size() && _setIds[newSelTab] == RecentStickerSetId && newSel >= newSelTab * emojiTabShift + _sets[newSelTab].size()) { if (newSel >= 0 && newSelTab < _setIds.size() && _setIds[newSelTab] == RecentStickerSetId && newSel >= newSelTab * MatrixRowShift + _sets[newSelTab].size()) {
xNewSel = newSel; xNewSel = newSel;
newSel -= _sets[newSelTab].size(); newSel -= _sets[newSelTab].size();
} }
@ -1627,7 +1627,7 @@ void StickerPanInner::updateSelected() {
bool StickerPanInner::animStep(float64 ms) { bool StickerPanInner::animStep(float64 ms) {
uint64 now = getms(); uint64 now = getms();
for (Animations::iterator i = _animations.begin(); i != _animations.end();) { for (Animations::iterator i = _animations.begin(); i != _animations.end();) {
int index = qAbs(i.key()) - 1, tab = (index / emojiTabShift), sel = index % emojiTabShift; int index = qAbs(i.key()) - 1, tab = (index / MatrixRowShift), sel = index % MatrixRowShift;
float64 dt = float64(now - i.value()) / st::emojiPanDuration; float64 dt = float64(now - i.value()) / st::emojiPanDuration;
if (dt >= 1) { if (dt >= 1) {
(index == SwitcherSelected ? _switcherHover : _hovers[tab][sel]) = (i.key() > 0) ? 1 : 0; (index == SwitcherSelected ? _switcherHover : _hovers[tab][sel]) = (i.key() > 0) ? 1 : 0;
@ -2403,7 +2403,7 @@ void EmojiPan::onDelayedHide() {
_removingSetId = 0; _removingSetId = 0;
} }
MentionsInner::MentionsInner(MentionsDropdown *parent, MentionRows *rows, HashtagRows *hrows) : _parent(parent), _rows(rows), _hrows(hrows), _sel(-1), _mouseSel(false), _overDelete(false) { MentionsInner::MentionsInner(MentionsDropdown *parent, MentionRows *rows, HashtagRows *hrows, BotCommandRows *crows) : _parent(parent), _rows(rows), _hrows(hrows), _crows(crows), _sel(-1), _mouseSel(false), _overDelete(false) {
} }
void MentionsInner::paintEvent(QPaintEvent *e) { void MentionsInner::paintEvent(QPaintEvent *e) {
@ -2413,28 +2413,26 @@ void MentionsInner::paintEvent(QPaintEvent *e) {
int32 availwidth = width() - 2 * st::mentionPadding.left() - st::mentionPhotoSize - 2 * st::mentionPadding.right(); int32 availwidth = width() - 2 * st::mentionPadding.left() - st::mentionPhotoSize - 2 * st::mentionPadding.right();
int32 htagleft = st::btnAttachPhoto.width + st::taMsgField.textMrg.left() - st::dlgShadow, htagwidth = width() - st::mentionPadding.right() - htagleft; int32 htagleft = st::btnAttachPhoto.width + st::taMsgField.textMrg.left() - st::dlgShadow, htagwidth = width() - st::mentionPadding.right() - htagleft;
int32 from = qFloor(e->rect().top() / st::mentionHeight), to = qFloor(e->rect().bottom() / st::mentionHeight) + 1, last = _rows->isEmpty() ? _hrows->size() : _rows->size(); int32 from = qFloor(e->rect().top() / st::mentionHeight), to = qFloor(e->rect().bottom() / st::mentionHeight) + 1;
int32 last = _rows->isEmpty() ? (_hrows->isEmpty() ? _crows->size() : _hrows->size()) : _rows->size();
bool hasUsername = _parent->filter().indexOf('@') > 1;
for (int32 i = from; i < to; ++i) { for (int32 i = from; i < to; ++i) {
if (i >= last) break; if (i >= last) break;
if (i == _sel) { if (i == _sel) {
p.fillRect(0, i * st::mentionHeight, width(), st::mentionHeight, st::dlgHoverBG->b); p.fillRect(0, i * st::mentionHeight, width(), st::mentionHeight, st::dlgHoverBG->b);
int skip = (st::mentionHeight - st::notifyClose.icon.pxHeight()) / 2; int skip = (st::mentionHeight - st::notifyClose.icon.pxHeight()) / 2;
if (_rows->isEmpty()) p.drawPixmap(QPoint(width() - st::notifyClose.icon.pxWidth() - skip, i * st::mentionHeight + skip), App::sprite(), st::notifyClose.icon); if (!_hrows->isEmpty()) p.drawPixmap(QPoint(width() - st::notifyClose.icon.pxWidth() - skip, i * st::mentionHeight + skip), App::sprite(), st::notifyClose.icon);
} }
p.setPen(st::black->p); p.setPen(st::black->p);
if (_rows->isEmpty()) { if (!_rows->isEmpty()) {
QString tag = st::mentionFont->m.elidedText('#' + _hrows->at(last - i - 1), Qt::ElideRight, htagwidth);
p.setFont(st::mentionFont->f);
p.drawText(htagleft, i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, tag);
} else {
UserData *user = _rows->at(last - i - 1); UserData *user = _rows->at(last - i - 1);
QString first = (_parent->filter().size() < 2) ? QString() : ('@' + user->username.mid(0, _parent->filter().size() - 1)), second = (_parent->filter().size() < 2) ? ('@' + user->username) : user->username.mid(_parent->filter().size() - 1); QString first = (_parent->filter().size() < 2) ? QString() : ('@' + user->username.mid(0, _parent->filter().size() - 1)), second = (_parent->filter().size() < 2) ? ('@' + user->username) : user->username.mid(_parent->filter().size() - 1);
int32 firstwidth = st::mentionFont->m.width(first), secondwidth = st::mentionFont->m.width(second), unamewidth = firstwidth + secondwidth, namewidth = user->nameText.maxWidth(); int32 firstwidth = st::mentionFont->m.width(first), secondwidth = st::mentionFont->m.width(second), unamewidth = firstwidth + secondwidth, namewidth = user->nameText.maxWidth();
if (availwidth < unamewidth + namewidth) { if (availwidth < unamewidth + namewidth) {
namewidth = (availwidth * namewidth) / (namewidth + unamewidth); namewidth = (availwidth * namewidth) / (namewidth + unamewidth);
unamewidth = availwidth - namewidth; unamewidth = availwidth - namewidth;
if (firstwidth <= unamewidth) { if (firstwidth < unamewidth + st::mentionFont->elidew) {
if (firstwidth < unamewidth) { if (firstwidth < unamewidth) {
first = st::mentionFont->m.elidedText(first, Qt::ElideRight, unamewidth); first = st::mentionFont->m.elidedText(first, Qt::ElideRight, unamewidth);
} else if (!second.isEmpty()) { } else if (!second.isEmpty()) {
@ -2448,14 +2446,96 @@ void MentionsInner::paintEvent(QPaintEvent *e) {
user->photo->load(); user->photo->load();
p.drawPixmap(st::mentionPadding.left(), i * st::mentionHeight + st::mentionPadding.top(), user->photo->pixRounded(st::mentionPhotoSize)); p.drawPixmap(st::mentionPadding.left(), i * st::mentionHeight + st::mentionPadding.top(), user->photo->pixRounded(st::mentionPhotoSize));
user->nameText.drawElided(p, 2 * st::mentionPadding.left() + st::mentionPhotoSize, i * st::mentionHeight + st::mentionTop, namewidth); user->nameText.drawElided(p, 2 * st::mentionPadding.left() + st::mentionPhotoSize, i * st::mentionHeight + st::mentionTop, namewidth);
p.setFont(st::mentionFont->f); p.setFont(st::mentionFont->f);
p.setPen(st::profileOnlineColor->p); p.setPen(st::profileOnlineColor->p);
p.drawText(2 * st::mentionPadding.left() + st::mentionPhotoSize + namewidth + st::mentionPadding.right(), i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, first); p.drawText(2 * st::mentionPadding.left() + st::mentionPhotoSize + namewidth + st::mentionPadding.right(), i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, first);
if (!second.isEmpty()) { if (!second.isEmpty()) {
p.setPen(st::profileOfflineColor->p); p.setPen(st::profileOfflineColor->p);
p.drawText(2 * st::mentionPadding.left() + st::mentionPhotoSize + namewidth + st::mentionPadding.right() + firstwidth, i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, second); p.drawText(2 * st::mentionPadding.left() + st::mentionPhotoSize + namewidth + st::mentionPadding.right() + firstwidth, i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, second);
} }
} else if (!_hrows->isEmpty()) {
QString first = (_parent->filter().size() < 2) ? QString() : ('#' + _hrows->at(last - i - 1).mid(0, _parent->filter().size() - 1)), second = (_parent->filter().size() < 2) ? ('#' + _hrows->at(last - i - 1)) : _hrows->at(last - i - 1).mid(_parent->filter().size() - 1);
int32 firstwidth = st::mentionFont->m.width(first), secondwidth = st::mentionFont->m.width(second);
if (htagwidth < firstwidth + secondwidth) {
if (htagwidth < firstwidth + st::mentionFont->elidew) {
first = st::mentionFont->m.elidedText(first + second, Qt::ElideRight, htagwidth);
second = QString();
} else {
second = st::mentionFont->m.elidedText(second, Qt::ElideRight, htagwidth - firstwidth);
}
}
p.setFont(st::mentionFont->f);
if (!first.isEmpty()) {
p.setPen(st::profileOnlineColor->p);
p.drawText(htagleft, i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, first);
}
if (!second.isEmpty()) {
p.setPen(st::profileOfflineColor->p);
p.drawText(htagleft + firstwidth, i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, second);
}
} else {
UserData *user = _crows->at(last - i - 1).first;
const BotCommand &command = _crows->at(last - i - 1).second;
QString toHighlight = command.command;
int32 botStatus = _parent->chat() ? _parent->chat()->botStatus : -1;
if (hasUsername || botStatus == 0 || botStatus == 2) {
toHighlight += '@' + user->username;
if (botStatus == 0 || botStatus == 2) {
user->photo->load();
p.drawPixmap(st::mentionPadding.left(), i * st::mentionHeight + st::mentionPadding.top(), user->photo->pixRounded(st::mentionPhotoSize));
}
}
int32 addleft = 0, widthleft = htagwidth;
QString first = (_parent->filter().size() < 2) ? QString() : ('/' + toHighlight.mid(0, _parent->filter().size() - 1)), second = (_parent->filter().size() < 2) ? ('/' + toHighlight) : toHighlight.mid(_parent->filter().size() - 1);
int32 firstwidth = st::botCommandFont->m.width(first), secondwidth = st::botCommandFont->m.width(second);
if (htagwidth < firstwidth + secondwidth) {
if (htagwidth < firstwidth + st::botCommandFont->elidew) {
first = st::botCommandFont->m.elidedText(first + second, Qt::ElideRight, htagwidth);
second = QString();
} else {
second = st::botCommandFont->m.elidedText(second, Qt::ElideRight, htagwidth - firstwidth);
}
}
p.setFont(st::botCommandFont->f);
if (!first.isEmpty()) {
p.setPen(st::profileOnlineColor->p);
p.drawText(htagleft, i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, first);
}
if (!second.isEmpty()) {
p.setPen(st::profileOfflineColor->p);
p.drawText(htagleft + firstwidth, i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, second);
}
addleft += firstwidth + secondwidth + st::mentionPadding.left();
widthleft -= firstwidth + secondwidth + st::mentionPadding.left();
QString params = command.params;
if (widthleft > st::mentionFont->elidew && !params.isEmpty()) {
p.setFont(st::mentionFont->f);
int32 paramswidth = st::mentionFont->m.width(params);
if (widthleft < paramswidth) {
params = st::mentionFont->m.elidedText(params, Qt::ElideRight, widthleft);
}
p.setPen(st::profileOfflineColor->p);
p.drawText(htagleft + addleft, i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, params);
addleft += paramswidth + st::mentionPadding.left();
widthleft -= paramswidth + st::mentionPadding.left();
}
QString description = command.description;
if (widthleft > st::botDescFont->elidew && !description.isEmpty()) {
p.setFont(st::botDescFont->f);
int32 descwidth = st::botDescFont->m.width(description);
if (widthleft < descwidth) {
description = st::botDescFont->m.elidedText(description, Qt::ElideRight, widthleft);
descwidth = st::botDescFont->m.width(description);
}
p.setPen(st::profileOfflineColor->p);
p.drawText(htagleft + addleft + (widthleft - descwidth), i * st::mentionHeight + st::mentionTop + st::mentionFont->ascent, description);
}
} }
} }
@ -2476,7 +2556,7 @@ void MentionsInner::clearSel() {
bool MentionsInner::moveSel(int direction) { bool MentionsInner::moveSel(int direction) {
_mouseSel = false; _mouseSel = false;
int32 maxSel = (_rows->isEmpty() ? _hrows->size() : _rows->size()); int32 maxSel = (_rows->isEmpty() ? (_hrows->isEmpty() ? _crows->size() : _hrows->size()) : _rows->size());
if (_sel >= maxSel || _sel < 0) { if (_sel >= maxSel || _sel < 0) {
if (direction < 0) setSel(maxSel - 1, true); if (direction < 0) setSel(maxSel - 1, true);
return (_sel >= 0 && _sel < maxSel); return (_sel >= 0 && _sel < maxSel);
@ -2488,9 +2568,23 @@ bool MentionsInner::moveSel(int direction) {
} }
bool MentionsInner::select() { bool MentionsInner::select() {
int32 maxSel = (_rows->isEmpty() ? _hrows->size() : _rows->size()); int32 maxSel = (_rows->isEmpty() ? (_hrows->isEmpty() ? _crows->size() : _hrows->size()) : _rows->size());
if (_sel >= 0 && _sel < maxSel) { if (_sel >= 0 && _sel < maxSel) {
QString result = _rows->isEmpty() ? ('#' + _hrows->at(_hrows->size() - _sel - 1)) : ('@' + _rows->at(_rows->size() - _sel - 1)->username); QString result;
if (!_rows->isEmpty()) {
result = '@' + _rows->at(_rows->size() - _sel - 1)->username;
} else if (!_hrows->isEmpty()) {
result = '#' + _hrows->at(_hrows->size() - _sel - 1);
} else {
UserData *user = _crows->at(_crows->size() - _sel - 1).first;
const BotCommand &command(_crows->at(_crows->size() - _sel - 1).second);
int32 botStatus = _parent->chat() ? _parent->chat()->botStatus : -1;
if (botStatus == 0 || botStatus == 2 || _parent->filter().indexOf('@') > 1) {
result = '/' + command.command + '@' + user->username;
} else {
result = '/' + command.command;
}
}
emit chosen(result); emit chosen(result);
return true; return true;
} }
@ -2542,7 +2636,7 @@ void MentionsInner::leaveEvent(QEvent *e) {
void MentionsInner::setSel(int sel, bool scroll) { void MentionsInner::setSel(int sel, bool scroll) {
_sel = sel; _sel = sel;
parentWidget()->update(); parentWidget()->update();
int32 maxSel = _rows->isEmpty() ? _hrows->size() : _rows->size(); int32 maxSel = _rows->isEmpty() ? (_hrows->isEmpty() ? _crows->size() : _hrows->size()) : _rows->size();
if (scroll && _sel >= 0 && _sel < maxSel) emit mustScrollTo(_sel * st::mentionHeight, (_sel + 1) * st::mentionHeight); if (scroll && _sel >= 0 && _sel < maxSel) emit mustScrollTo(_sel * st::mentionHeight, (_sel + 1) * st::mentionHeight);
} }
@ -2552,7 +2646,7 @@ void MentionsInner::onUpdateSelected(bool force) {
int w = width(), mouseY = mouse.y(); int w = width(), mouseY = mouse.y();
_overDelete = _rows->isEmpty() && (mouse.x() >= w - st::mentionHeight); _overDelete = _rows->isEmpty() && (mouse.x() >= w - st::mentionHeight);
int32 sel = mouseY / int32(st::mentionHeight), maxSel = _rows->isEmpty() ? _hrows->size() : _rows->size(); int32 sel = mouseY / int32(st::mentionHeight), maxSel = _rows->isEmpty() ? (_hrows->isEmpty() ? _crows->size() : _hrows->size()) : _rows->size();
if (sel < 0 || sel >= maxSel) { if (sel < 0 || sel >= maxSel) {
sel = -1; sel = -1;
} }
@ -2570,7 +2664,7 @@ void MentionsInner::onParentGeometryChanged() {
} }
MentionsDropdown::MentionsDropdown(QWidget *parent) : QWidget(parent), MentionsDropdown::MentionsDropdown(QWidget *parent) : QWidget(parent),
_scroll(this, st::mentionScroll), _inner(this, &_rows, &_hrows), _chat(0), _hiding(false), a_opacity(0), _shadow(st::dropdownDef.shadow) { _scroll(this, st::mentionScroll), _inner(this, &_rows, &_hrows, &_crows), _chat(0), _hiding(false), a_opacity(0), _shadow(st::dropdownDef.shadow) {
_hideTimer.setSingleShot(true); _hideTimer.setSingleShot(true);
connect(&_hideTimer, SIGNAL(timeout()), this, SLOT(hideStart())); connect(&_hideTimer, SIGNAL(timeout()), this, SLOT(hideStart()));
connect(&_inner, SIGNAL(chosen(QString)), this, SIGNAL(chosen(QString))); connect(&_inner, SIGNAL(chosen(QString)), this, SIGNAL(chosen(QString)));
@ -2608,8 +2702,9 @@ void MentionsDropdown::paintEvent(QPaintEvent *e) {
} }
void MentionsDropdown::showFiltered(ChatData *chat, QString start) { void MentionsDropdown::showFiltered(PeerData *peer, QString start) {
_chat = chat; _chat = peer->chat ? peer->asChat() : 0;
_user = peer->chat ? 0 : peer->asUser();
start = start.toLower(); start = start.toLower();
bool toDown = (_filter != start); bool toDown = (_filter != start);
if (toDown) { if (toDown) {
@ -2621,10 +2716,11 @@ void MentionsDropdown::showFiltered(ChatData *chat, QString start) {
void MentionsDropdown::updateFiltered(bool toDown) { void MentionsDropdown::updateFiltered(bool toDown) {
int32 now = unixtime(); int32 now = unixtime();
QMultiMap<int32, UserData*> ordered;
MentionRows rows; MentionRows rows;
HashtagRows hrows; HashtagRows hrows;
BotCommandRows crows;
if (_filter.at(0) == '@') { if (_filter.at(0) == '@') {
QMultiMap<int32, UserData*> ordered;
rows.reserve(_chat->participants.isEmpty() ? _chat->lastAuthors.size() : _chat->participants.size()); rows.reserve(_chat->participants.isEmpty() ? _chat->lastAuthors.size() : _chat->participants.size());
if (_chat->participants.isEmpty()) { if (_chat->participants.isEmpty()) {
if (_chat->count > 0) { if (_chat->count > 0) {
@ -2653,23 +2749,79 @@ void MentionsDropdown::updateFiltered(bool toDown) {
rows.push_back(i.value()); rows.push_back(i.value());
} }
} }
} else { } else if (_filter.at(0) == '#') {
const RecentHashtagPack &recent(cRecentWriteHashtags()); const RecentHashtagPack &recent(cRecentWriteHashtags());
hrows.reserve(recent.size()); hrows.reserve(recent.size());
for (RecentHashtagPack::const_iterator i = recent.cbegin(), e = recent.cend(); i != e; ++i) { for (RecentHashtagPack::const_iterator i = recent.cbegin(), e = recent.cend(); i != e; ++i) {
if (_filter.size() > 1 && (!i->first.startsWith(_filter.midRef(1), Qt::CaseInsensitive) || i->first.size() + 1 == _filter.size())) continue; if (_filter.size() > 1 && (!i->first.startsWith(_filter.midRef(1), Qt::CaseInsensitive) || i->first.size() + 1 == _filter.size())) continue;
hrows.push_back(i->first); hrows.push_back(i->first);
} }
} else if (_filter.at(0) == '/') {
bool hasUsername = _filter.indexOf('@') > 1;
QMap<UserData*, bool> bots;
int32 cnt = 0;
if (_chat) {
if (_chat->participants.isEmpty()) {
if (_chat->count > 0) {
App::api()->requestFullPeer(_chat);
}
} else {
int32 index = 0;
for (ChatData::Participants::const_iterator i = _chat->participants.cbegin(), e = _chat->participants.cend(); i != e; ++i) {
UserData *user = i.key();
if (!user->botInfo || user->botInfo->commands.isEmpty()) continue;
bots.insert(user, true);
cnt += user->botInfo->commands.size();
}
}
} else if (_user->botInfo) {
cnt = _user->botInfo->commands.size();
bots.insert(_user, true);
}
if (cnt) {
crows.reserve(cnt);
int32 botStatus = _chat ? _chat->botStatus : -1;
if (_chat) {
for (MentionRows::const_iterator i = _chat->lastAuthors.cbegin(), e = _chat->lastAuthors.cend(); i != e; ++i) {
UserData *user = *i;
if (!user->botInfo || user->botInfo->commands.isEmpty()) continue;
for (int32 j = 0, l = user->botInfo->commands.size(); j < l; ++j) {
if (_filter.size() > 1) {
QString toFilter = (hasUsername || botStatus == 0 || botStatus == 2) ? user->botInfo->commands.at(j).command + '@' + user->username : user->botInfo->commands.at(j).command;
if (!toFilter.startsWith(_filter.midRef(1), Qt::CaseInsensitive) || toFilter.size() + 1 == _filter.size()) continue;
}
crows.push_back(qMakePair(user, user->botInfo->commands.at(j)));
}
if (!bots.isEmpty()) {
bots.remove(user);
}
}
}
if (!bots.isEmpty()) {
for (QMap<UserData*, bool>::const_iterator i = bots.cbegin(), e = bots.cend(); i != e; ++i) {
UserData *user = i.key();
for (int32 j = 0, l = user->botInfo->commands.size(); j < l; ++j) {
if (_filter.size() > 1) {
QString toFilter = (hasUsername || botStatus == 0 || botStatus == 2) ? user->botInfo->commands.at(j).command + '@' + user->username : user->botInfo->commands.at(j).command;
if (!toFilter.startsWith(_filter.midRef(1), Qt::CaseInsensitive) || toFilter.size() + 1 == _filter.size()) continue;
}
crows.push_back(qMakePair(user, user->botInfo->commands.at(j)));
}
}
}
}
} }
if (rows.isEmpty() && hrows.isEmpty()) { if (rows.isEmpty() && hrows.isEmpty() && crows.isEmpty()) {
if (!isHidden()) { if (!isHidden()) {
hideStart(); hideStart();
_rows.clear(); _rows.clear();
_hrows.clear(); _hrows.clear();
_crows.clear();
} }
} else { } else {
_rows = rows; _rows = rows;
_hrows = hrows; _hrows = hrows;
_crows = crows;
bool hidden = _hiding || isHidden(); bool hidden = _hiding || isHidden();
if (hidden) { if (hidden) {
show(); show();
@ -2692,7 +2844,7 @@ void MentionsDropdown::setBoundings(QRect boundings) {
} }
void MentionsDropdown::recount(bool toDown) { void MentionsDropdown::recount(bool toDown) {
int32 h = (_rows.isEmpty() ? _hrows.size() : _rows.size()) * st::mentionHeight, oldst = _scroll.scrollTop(), st = oldst; int32 h = (_rows.isEmpty() ? (_hrows.isEmpty() ? _crows.size() : _hrows.size()) : _rows.size()) * st::mentionHeight, oldst = _scroll.scrollTop(), st = oldst;
if (_inner.height() != h) { if (_inner.height() != h) {
st += h - _inner.height(); st += h - _inner.height();
@ -2780,6 +2932,10 @@ const QString &MentionsDropdown::filter() const {
return _filter; return _filter;
} }
ChatData *MentionsDropdown::chat() const {
return _chat;
}
int32 MentionsDropdown::innerTop() { int32 MentionsDropdown::innerTop() {
return _scroll.scrollTop(); return _scroll.scrollTop();
} }

View File

@ -467,6 +467,7 @@ private:
typedef QList<UserData*> MentionRows; typedef QList<UserData*> MentionRows;
typedef QList<QString> HashtagRows; typedef QList<QString> HashtagRows;
typedef QList<QPair<UserData*, BotCommand> > BotCommandRows;
class MentionsDropdown; class MentionsDropdown;
class MentionsInner : public QWidget { class MentionsInner : public QWidget {
@ -474,7 +475,7 @@ class MentionsInner : public QWidget {
public: public:
MentionsInner(MentionsDropdown *parent, MentionRows *rows, HashtagRows *hrows); MentionsInner(MentionsDropdown *parent, MentionRows *rows, HashtagRows *hrows, BotCommandRows *crows);
void paintEvent(QPaintEvent *e); void paintEvent(QPaintEvent *e);
@ -505,6 +506,7 @@ private:
MentionsDropdown *_parent; MentionsDropdown *_parent;
MentionRows *_rows; MentionRows *_rows;
HashtagRows *_hrows; HashtagRows *_hrows;
BotCommandRows *_crows;
int32 _sel; int32 _sel;
bool _mouseSel; bool _mouseSel;
QPoint _mousePos; QPoint _mousePos;
@ -523,13 +525,14 @@ public:
void fastHide(); void fastHide();
void showFiltered(ChatData *chat, QString start); void showFiltered(PeerData *peer, QString start);
void updateFiltered(bool toDown = false); void updateFiltered(bool toDown = false);
void setBoundings(QRect boundings); void setBoundings(QRect boundings);
bool animStep(float64 ms); bool animStep(float64 ms);
const QString &filter() const; const QString &filter() const;
ChatData *chat() const;
int32 innerTop(); int32 innerTop();
int32 innerBottom(); int32 innerBottom();
@ -556,11 +559,13 @@ private:
QPixmap _cache; QPixmap _cache;
MentionRows _rows; MentionRows _rows;
HashtagRows _hrows; HashtagRows _hrows;
BotCommandRows _crows;
ScrollArea _scroll; ScrollArea _scroll;
MentionsInner _inner; MentionsInner _inner;
ChatData *_chat; ChatData *_chat;
UserData *_user;
QString _filter; QString _filter;
QRect _boundings; QRect _boundings;

View File

@ -179,7 +179,7 @@ EmojiPtr FlatTextarea::getSingleEmoji() const {
return 0; return 0;
} }
void FlatTextarea::getMentionHashtagStart(QString &start) const { void FlatTextarea::getMentionHashtagBotCommandStart(QString &start) const {
int32 pos = textCursor().position(); int32 pos = textCursor().position();
if (textCursor().anchor() != pos) return; if (textCursor().anchor() != pos) return;
@ -195,11 +195,16 @@ void FlatTextarea::getMentionHashtagStart(QString &start) const {
QTextCharFormat f = fr.charFormat(); QTextCharFormat f = fr.charFormat();
if (f.isImageFormat()) continue; if (f.isImageFormat()) continue;
bool mentionInCommand = false;
QString t(fr.text()); QString t(fr.text());
for (int i = pos - p; i > 0; --i) { for (int i = pos - p; i > 0; --i) {
if (t.at(i - 1) == '@') { if (t.at(i - 1) == '@') {
if ((pos - p - i < 1 || t.at(i).isLetter()) && (i < 2 || !(t.at(i - 2).isLetterOrNumber() || t.at(i - 2) == '_'))) { if ((pos - p - i < 1 || t.at(i).isLetter()) && (i < 2 || !(t.at(i - 2).isLetterOrNumber() || t.at(i - 2) == '_'))) {
start = t.mid(i - 1, pos - p - i + 1); start = t.mid(i - 1, pos - p - i + 1);
} else if ((pos - p - i < 1 || t.at(i).isLetter()) && i > 2 && (t.at(i - 2).isLetterOrNumber() || t.at(i - 2) == '_') && !mentionInCommand) {
mentionInCommand = true;
--i;
continue;
} }
return; return;
} else if (t.at(i - 1) == '#') { } else if (t.at(i - 1) == '#') {
@ -207,15 +212,20 @@ void FlatTextarea::getMentionHashtagStart(QString &start) const {
start = t.mid(i - 1, pos - p - i + 1); start = t.mid(i - 1, pos - p - i + 1);
} }
return; return;
} else if (t.at(i - 1) == '/') {
if (i < 2) {
start = t.mid(i - 1, pos - p - i + 1);
}
return;
} }
if (pos - p - i > 63) break; if (pos - p - i > 127 || (!mentionInCommand && (pos - p - i > 63))) break;
if (!t.at(i - 1).isLetterOrNumber() && t.at(i - 1) != '_') break; if (!t.at(i - 1).isLetterOrNumber() && t.at(i - 1) != '_') break;
} }
return; return;
} }
} }
void FlatTextarea::onMentionOrHashtagInsert(QString mentionOrHashtag) { void FlatTextarea::onMentionHashtagOrBotCommandInsert(QString str) {
QTextCursor c(textCursor()); QTextCursor c(textCursor());
int32 pos = c.position(); int32 pos = c.position();
@ -231,31 +241,37 @@ void FlatTextarea::onMentionOrHashtagInsert(QString mentionOrHashtag) {
QTextCharFormat f = fr.charFormat(); QTextCharFormat f = fr.charFormat();
if (f.isImageFormat()) continue; if (f.isImageFormat()) continue;
bool mentionInCommand = false;
QString t(fr.text()); QString t(fr.text());
for (int i = pos - p; i > 0; --i) { for (int i = pos - p; i > 0; --i) {
if (t.at(i - 1) == '@' || t.at(i - 1) == '#') { if (t.at(i - 1) == '@' || t.at(i - 1) == '#' || t.at(i - 1) == '/') {
if ((i == pos - p || t.at(i).isLetter() || t.at(i - 1) == '#') && (i < 2 || !(t.at(i - 2).isLetterOrNumber() || t.at(i - 2) == '_'))) { if ((i == pos - p || t.at(i).isLetter() || t.at(i - 1) == '#') && (i < 2 || !(t.at(i - 2).isLetterOrNumber() || t.at(i - 2) == '_'))) {
c.setPosition(p + i - 1, QTextCursor::MoveAnchor); c.setPosition(p + i - 1, QTextCursor::MoveAnchor);
int till = p + i; int till = p + i;
for (; (till < e) && (till - p - i + 1 < mentionOrHashtag.size()); ++till) { for (; (till < e) && (till - p - i + 1 < str.size()); ++till) {
if (t.at(till - p).toLower() != mentionOrHashtag.at(till - p - i + 1).toLower()) { if (t.at(till - p).toLower() != str.at(till - p - i + 1).toLower()) {
break; break;
} }
} }
if (till - p - i + 1 == mentionOrHashtag.size() && till < e && t.at(till - p) == ' ') { if (till - p - i + 1 == str.size() && till < e && t.at(till - p) == ' ') {
++till; ++till;
} }
c.setPosition(till, QTextCursor::KeepAnchor); c.setPosition(till, QTextCursor::KeepAnchor);
c.insertText(mentionOrHashtag + ' '); c.insertText(str + ' ');
return; return;
} else if ((i == pos - p || t.at(i).isLetter()) && t.at(i - 1) == '@' && i > 2 && (t.at(i - 2).isLetterOrNumber() || t.at(i - 2) == '_') && !mentionInCommand) {
mentionInCommand = true;
--i;
continue;
} }
break; break;
} }
if (pos - p - i > 63) break; if (pos - p - i > 127 || (!mentionInCommand && (pos - p - i > 63))) break;
if (!t.at(i - 1).isLetterOrNumber() && t.at(i - 1) != '_') break; if (!t.at(i - 1).isLetterOrNumber() && t.at(i - 1) != '_') break;
} }
break;
} }
c.insertText(mentionOrHashtag + ' '); c.insertText(str + ' ');
} }
void FlatTextarea::getSingleEmojiFragment(QString &text, QTextFragment &fragment) const { void FlatTextarea::getSingleEmojiFragment(QString &text, QTextFragment &fragment) const {

View File

@ -49,7 +49,7 @@ public:
QSize minimumSizeHint() const; QSize minimumSizeHint() const;
EmojiPtr getSingleEmoji() const; EmojiPtr getSingleEmoji() const;
void getMentionHashtagStart(QString &start) const; void getMentionHashtagBotCommandStart(QString &start) const;
void removeSingleEmoji(); void removeSingleEmoji();
QString getText(int32 start = 0, int32 end = -1) const; QString getText(int32 start = 0, int32 end = -1) const;
bool hasText() const; bool hasText() const;
@ -72,7 +72,7 @@ public slots:
void onUndoAvailable(bool avail); void onUndoAvailable(bool avail);
void onRedoAvailable(bool avail); void onRedoAvailable(bool avail);
void onMentionOrHashtagInsert(QString mentionOrHashtag); void onMentionHashtagOrBotCommandInsert(QString str);
signals: signals:

View File

@ -661,3 +661,7 @@ void ScrollArea::updateColors(const style::color &bar, const style::color &bg, c
hor.update(); hor.update();
vert.update(); vert.update();
} }
ScrollArea::~ScrollArea() {
takeWidget();
}

View File

@ -131,6 +131,8 @@ public:
void updateColors(const style::color &bar, const style::color &bg, const style::color &barOver, const style::color &bgOver); void updateColors(const style::color &bar, const style::color &bg, const style::color &barOver, const style::color &bgOver);
~ScrollArea();
public slots: public slots:
void scrollToY(int toTop, int toBottom = -1); void scrollToY(int toTop, int toBottom = -1);

View File

@ -30,7 +30,7 @@ namespace {
const QRegularExpression _reMailStart(qsl("^[a-zA-Z\\-_\\.0-9]{1,256}\\@")); const QRegularExpression _reMailStart(qsl("^[a-zA-Z\\-_\\.0-9]{1,256}\\@"));
const QRegularExpression _reHashtag(qsl("(^|[\\s\\.,:;<>|'\"\\[\\]\\{\\}`\\~\\!\\%\\^\\*\\(\\)\\-\\+=\\x10])#[\\w]{2,64}([\\W]|$)"), QRegularExpression::UseUnicodePropertiesOption); const QRegularExpression _reHashtag(qsl("(^|[\\s\\.,:;<>|'\"\\[\\]\\{\\}`\\~\\!\\%\\^\\*\\(\\)\\-\\+=\\x10])#[\\w]{2,64}([\\W]|$)"), QRegularExpression::UseUnicodePropertiesOption);
const QRegularExpression _reMention(qsl("(^|[\\s\\.,:;<>|'\"\\[\\]\\{\\}`\\~\\!\\%\\^\\*\\(\\)\\-\\+=\\x10])@[A-Za-z_0-9]{5,32}([\\W]|$)"), QRegularExpression::UseUnicodePropertiesOption); const QRegularExpression _reMention(qsl("(^|[\\s\\.,:;<>|'\"\\[\\]\\{\\}`\\~\\!\\%\\^\\*\\(\\)\\-\\+=\\x10])@[A-Za-z_0-9]{5,32}([\\W]|$)"), QRegularExpression::UseUnicodePropertiesOption);
const QRegularExpression _reBotCommand(qsl("(^|[\\s\\.,:;<>|'\"\\[\\]\\{\\}`\\~\\!\\%\\^\\*\\(\\)\\-\\+=\\x10])/[\\w]{1,64}([\\W]|$)"), QRegularExpression::UseUnicodePropertiesOption); const QRegularExpression _reBotCommand(qsl("(^|[\\s\\.,:;<>|'\"\\[\\]\\{\\}`\\~\\!\\%\\^\\*\\(\\)\\-\\+=\\x10])/[\\w]{1,64}(@[A-Za-z_0-9]{5,32})?([\\W]|$)"), QRegularExpression::UseUnicodePropertiesOption);
QSet<int32> _validProtocols, _validTopDomains; QSet<int32> _validProtocols, _validTopDomains;
const style::textStyle *_textStyle = 0; const style::textStyle *_textStyle = 0;
@ -1104,9 +1104,33 @@ public:
if (_yTo >= 0 && _y + _yDelta >= _yTo) return false; if (_yTo >= 0 && _y + _yDelta >= _yTo) return false;
if (_y + _yDelta + _fontHeight <= _yFrom) return true; if (_y + _yDelta + _fontHeight <= _yFrom) return true;
uint16 trimmedLineEnd = _lineEnd;
for (; trimmedLineEnd > _lineStart; --trimmedLineEnd) {
QChar ch = _t->_text.at(trimmedLineEnd - 1);
if ((ch != QChar::Space || trimmedLineEnd == _lineStart + 1) && ch != QChar::LineFeed) {
break;
}
}
ITextBlock *_endBlock = (_endBlockIter == _end) ? 0 : (*_endBlockIter); ITextBlock *_endBlock = (_endBlockIter == _end) ? 0 : (*_endBlockIter);
bool elidedLine = _elideLast && _endBlock && (_y + _lineHeight >= _yTo); bool elidedLine = _elideLast && _endBlock && (_y + _lineHeight >= _yTo);
int blockIndex = _lineStartBlock;
ITextBlock *currentBlock = _t->_blocks[blockIndex];
ITextBlock *nextBlock = (++blockIndex < _blocksSize) ? _t->_blocks[blockIndex] : 0;
int32 delta = (currentBlock->from() < _lineStart ? qMin(_lineStart - currentBlock->from(), 2) : 0);
_localFrom = _lineStart - delta;
int32 lineEnd = (_endBlock && _endBlock->from() < trimmedLineEnd && !elidedLine) ? qMin(uint16(trimmedLineEnd + 2), _blockEnd(_t, _endBlockIter, _end)) : trimmedLineEnd;
QString lineText = _t->_text.mid(_localFrom, lineEnd - _localFrom);
int32 lineStart = delta, lineLength = trimmedLineEnd - _lineStart;
if (elidedLine) {
initParagraphBidi();
prepareElidedLine(lineText, lineStart, lineLength, _endBlock);
}
QFixed x = _x; QFixed x = _x;
if (_align & Qt::AlignHCenter) { if (_align & Qt::AlignHCenter) {
x += (_wLeft / 2).toInt(); x += (_wLeft / 2).toInt();
@ -1154,34 +1178,9 @@ public:
} }
} }
/* // lpadding is counted to _wLeft if (trimmedLineEnd == _lineStart && !elidedLine) return true;
for (; _lineStart < _lineEnd; ++_lineStart) {
if (_t->_text.at(_lineStart) != QChar::Space) {
break;
}
}/**/
for (; _lineEnd > _lineStart; --_lineEnd) {
QChar ch = _t->_text.at(_lineEnd - 1);
if ((ch != QChar::Space || _lineEnd == _lineStart + 1) && ch != QChar::LineFeed) {
break;
}
}/**/
if (_lineEnd == _lineStart && !elidedLine) return true;
initParagraphBidi(); // if was not inited if (!elidedLine) initParagraphBidi(); // if was not inited
int blockIndex = _lineStartBlock;
ITextBlock *currentBlock = _t->_blocks[blockIndex];
ITextBlock *nextBlock = (++blockIndex < _blocksSize) ? _t->_blocks[blockIndex] : 0;
int32 delta = (currentBlock->from() < _lineStart ? qMin(_lineStart - currentBlock->from(), 2) : 0);
_localFrom = _lineStart - delta;
int32 lineEnd = (_endBlock && _endBlock->from() < _lineEnd && !elidedLine) ? qMin(uint16(_lineEnd + 2), _blockEnd(_t, _endBlockIter, _end)) : _lineEnd;
QString lineText = _t->_text.mid(_localFrom, lineEnd - _localFrom);
int32 lineStart = delta, lineLength = _lineEnd - _lineStart;
if (elidedLine) prepareElidedLine(lineText, lineStart, lineLength, _endBlock);
_f = _t->_font; _f = _t->_font;
QStackTextEngine engine(lineText, _f->f); QStackTextEngine engine(lineText, _f->f);
@ -1218,7 +1217,7 @@ public:
} }
if (si.analysis.flags == QScriptAnalysis::Object) { if (si.analysis.flags == QScriptAnalysis::Object) {
if (_type == TextBlockEmoji || _type == TextBlockSkip) { if (_type == TextBlockEmoji || _type == TextBlockSkip) {
si.width = currentBlock->f_width() + (nextBlock == _endBlock && (!nextBlock || nextBlock->from() >= _lineEnd) ? 0 : currentBlock->f_rpadding()); si.width = currentBlock->f_width() + (nextBlock == _endBlock && (!nextBlock || nextBlock->from() >= trimmedLineEnd) ? 0 : currentBlock->f_rpadding());
} }
} }
} }
@ -1266,8 +1265,8 @@ public:
*_getSymbolAfter = false; *_getSymbolAfter = false;
*_getSymbolUpon = false; *_getSymbolUpon = false;
} else { } else {
*_getSymbol = (_lineEnd > _lineStart) ? (_lineEnd - 1) : _lineStart; *_getSymbol = (trimmedLineEnd > _lineStart) ? (trimmedLineEnd - 1) : _lineStart;
*_getSymbolAfter = (_lineEnd > _lineStart) ? true : false; *_getSymbolAfter = (trimmedLineEnd > _lineStart) ? true : false;
*_getSymbolUpon = false; *_getSymbolUpon = false;
} }
return false; return false;
@ -4152,7 +4151,7 @@ LinkRanges textParseLinks(const QString &text, int32 flags, bool rich) { // some
if (!mBotCommand.capturedRef(1).isEmpty()) { if (!mBotCommand.capturedRef(1).isEmpty()) {
++botCommandOffset; ++botCommandOffset;
} }
if (!mBotCommand.capturedRef(2).isEmpty()) { if (!mBotCommand.capturedRef(3).isEmpty()) {
--botCommandEnd; --botCommandEnd;
} }
} }

View File

@ -435,8 +435,7 @@ struct TextParseOptions {
int32 maxh; int32 maxh;
Qt::LayoutDirection dir; Qt::LayoutDirection dir;
}; };
extern const TextParseOptions _defaultOptions; extern const TextParseOptions _defaultOptions, _textPlainOptions;
extern const TextParseOptions _textPlainOptions;
enum TextSelectType { enum TextSelectType {
TextSelectLetters = 0x01, TextSelectLetters = 0x01,

View File

@ -109,6 +109,14 @@ namespace {
inline const HistoryForwarded *toHistoryForwarded(const HistoryItem *item) { inline const HistoryForwarded *toHistoryForwarded(const HistoryItem *item) {
return item ? item->toHistoryForwarded() : 0; return item ? item->toHistoryForwarded() : 0;
} }
inline const TextParseOptions &itemTextParseOptions(HistoryItem *item) {
History *h = item->history();
UserData *f = item->from();
if ((!h->peer->chat && h->peer->asUser()->botInfo) || (!f->chat && f->asUser()->botInfo) || (h->peer->chat && h->peer->asChat()->botStatus >= 0)) {
return _historyBotOptions;
}
return _historyTextOptions;
}
} }
void historyInit() { void historyInit() {
@ -155,7 +163,7 @@ void DialogRow::paint(QPainter &p, int32 w, bool act, bool sel) const {
rectForName.setLeft(rectForName.left() + st::dlgChatImgSkip); rectForName.setLeft(rectForName.left() + st::dlgChatImgSkip);
} }
HistoryItem *last = history->last; HistoryItem *last = history->lastMsg;
if (!last) { if (!last) {
p.setFont(st::dlgHistFont->f); p.setFont(st::dlgHistFont->f);
p.setPen((act ? st::dlgActiveColor : st::dlgSystemColor)->p); p.setPen((act ? st::dlgActiveColor : st::dlgSystemColor)->p);
@ -296,9 +304,11 @@ History::History(const PeerId &peerId) : width(0), height(0)
, peer(App::peer(peerId)) , peer(App::peer(peerId))
, oldLoaded(false) , oldLoaded(false)
, newLoaded(true) , newLoaded(true)
, last(0) , lastMsg(0)
, activeMsgId(0) , activeMsgId(0)
, draftToId(0) , draftToId(0)
, lastKeyboardId(0)
, lastKeyboardFrom(0)
, lastWidth(0) , lastWidth(0)
, lastScrollTop(History::ScrollMax) , lastScrollTop(History::ScrollMax)
, mute(isNotifyMuted(peer->notify)) , mute(isNotifyMuted(peer->notify))
@ -377,7 +387,7 @@ bool DialogsList::del(const PeerId &peerId, DialogRow *replacedBy) {
} }
void DialogsIndexed::peerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars) { void DialogsIndexed::peerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars) {
if (byName) { if (sortMode == DialogsSortByName) {
DialogRow *mainRow = list.adjustByName(peer); DialogRow *mainRow = list.adjustByName(peer);
if (!mainRow) return; if (!mainRow) return;
@ -406,7 +416,7 @@ void DialogsIndexed::peerNameChanged(PeerData *peer, const PeerData::Names &oldN
for (PeerData::NameFirstChars::const_iterator i = toAdd.cbegin(), e = toAdd.cend(); i != e; ++i) { for (PeerData::NameFirstChars::const_iterator i = toAdd.cbegin(), e = toAdd.cend(); i != e; ++i) {
DialogsIndex::iterator j = index.find(*i); DialogsIndex::iterator j = index.find(*i);
if (j == index.cend()) { if (j == index.cend()) {
j = index.insert(*i, new DialogsList(byName)); j = index.insert(*i, new DialogsList(sortMode));
} }
j.value()->addByName(history); j.value()->addByName(history);
} }
@ -428,7 +438,7 @@ void DialogsIndexed::peerNameChanged(PeerData *peer, const PeerData::Names &oldN
} }
} }
for (PeerData::NameFirstChars::const_iterator i = toRemove.cbegin(), e = toRemove.cend(); i != e; ++i) { for (PeerData::NameFirstChars::const_iterator i = toRemove.cbegin(), e = toRemove.cend(); i != e; ++i) {
history->dialogs.remove(*i); if (sortMode == DialogsSortByDate) history->dialogs.remove(*i);
DialogsIndex::iterator j = index.find(*i); DialogsIndex::iterator j = index.find(*i);
if (j != index.cend()) { if (j != index.cend()) {
j.value()->del(peer->id, mainRow); j.value()->del(peer->id, mainRow);
@ -437,9 +447,13 @@ void DialogsIndexed::peerNameChanged(PeerData *peer, const PeerData::Names &oldN
for (PeerData::NameFirstChars::const_iterator i = toAdd.cbegin(), e = toAdd.cend(); i != e; ++i) { for (PeerData::NameFirstChars::const_iterator i = toAdd.cbegin(), e = toAdd.cend(); i != e; ++i) {
DialogsIndex::iterator j = index.find(*i); DialogsIndex::iterator j = index.find(*i);
if (j == index.cend()) { if (j == index.cend()) {
j = index.insert(*i, new DialogsList(byName)); j = index.insert(*i, new DialogsList(sortMode));
}
if (sortMode == DialogsSortByDate) {
history->dialogs.insert(*i, j.value()->addByPos(history));
} else {
j.value()->addToEnd(history);
} }
history->dialogs.insert(*i, j.value()->addByPos(history));
} }
} }
} }
@ -535,7 +549,7 @@ HistoryItem *Histories::addToBack(const MTPmessage &msg, int msgState) {
if (!h.value()->loadedAtBottom()) { if (!h.value()->loadedAtBottom()) {
HistoryItem *item = h.value()->addToHistory(msg); HistoryItem *item = h.value()->addToHistory(msg);
if (item) { if (item) {
h.value()->last = item; h.value()->lastMsg = item;
if (msgState > 0) { if (msgState > 0) {
h.value()->newItemAdded(item); h.value()->newItemAdded(item);
} }
@ -586,6 +600,9 @@ HistoryItem *History::createItem(HistoryBlock *block, const MTPmessage &msg, boo
} else { } else {
result = new HistoryMessage(this, block, msg.c_message()); result = new HistoryMessage(this, block, msg.c_message());
} }
if (msg.c_message().has_reply_markup()) {
App::feedReplyMarkup(msgId, msg.c_message().vreply_markup);
}
break; break;
case mtpc_messageService: { case mtpc_messageService: {
@ -762,7 +779,7 @@ HistoryItem *History::doAddToBack(HistoryBlock *to, bool newBlock, HistoryItem *
} }
} }
to->push_back(adding); to->push_back(adding);
last = adding; lastMsg = adding;
adding->y = to->height; adding->y = to->height;
if (width) { if (width) {
int32 dh = adding->resize(width); int32 dh = adding->resize(width);
@ -785,12 +802,23 @@ HistoryItem *History::doAddToBack(HistoryBlock *to, bool newBlock, HistoryItem *
} }
} }
} }
if (peer->chat && adding->from()->id) { if (adding->from()->id) {
QList<UserData*> *lastAuthors = &(peer->asChat()->lastAuthors); if (peer->chat) {
int prev = lastAuthors->indexOf(adding->from()); QList<UserData*> *lastAuthors = &(peer->asChat()->lastAuthors);
if (prev > 0) { int prev = lastAuthors->indexOf(adding->from());
lastAuthors->removeAt(prev); if (prev > 0) {
lastAuthors->push_front(adding->from()); lastAuthors->removeAt(prev);
}
if (prev) {
lastAuthors->push_front(adding->from());
}
}
if (adding->hasReplyMarkup()) {
lastKeyboardId = adding->id;
lastKeyboardFrom = adding->from()->id;
} else if (lastKeyboardFrom == adding->from()->id) {
lastKeyboardId = 0;
lastKeyboardFrom = 0;
} }
} }
return adding; return adding;
@ -894,7 +922,15 @@ void History::addToFront(const QVector<MTPMessage> &slice) {
} }
} }
} }
if (lastAuthors && item->from()->id && !lastAuthors->contains(item->from())) lastAuthors->push_back(item->from()); if (item->from()->id) {
if (lastAuthors && !lastAuthors->contains(item->from())) {
if (item->hasReplyMarkup() && !lastKeyboardId) {
lastKeyboardId = item->id;
lastKeyboardFrom = item->from()->id;
}
lastAuthors->push_back(item->from());
}
}
} }
if (App::wnd()) App::wnd()->mediaOverviewUpdated(peer); if (App::wnd()) App::wnd()->mediaOverviewUpdated(peer);
} }
@ -1143,9 +1179,9 @@ void History::fixLastMessage(bool wasAtBottom) {
wasAtBottom = false; wasAtBottom = false;
} }
if (wasAtBottom) { if (wasAtBottom) {
last = back()->back(); lastMsg = back()->back();
} else { } else {
last = 0; lastMsg = 0;
if (App::main()) { if (App::main()) {
App::main()->checkPeerHistory(peer); App::main()->checkPeerHistory(peer);
} }
@ -1161,12 +1197,12 @@ void History::loadAround(MsgId msgId) {
if (!item || !item->block()) { if (!item || !item->block()) {
clear(true); clear(true);
} }
newLoaded = last && !last->detached(); newLoaded = lastMsg && !lastMsg->detached();
} else { } else {
if (!loadedAtBottom()) { if (!loadedAtBottom()) {
clear(true); clear(true);
} }
newLoaded = isEmpty() || (last && !last->detached()); newLoaded = isEmpty() || (lastMsg && !lastMsg->detached());
} }
} }
} }
@ -1252,7 +1288,7 @@ void History::clear(bool leaveItems) {
setMsgCount(0); setMsgCount(0);
if (!leaveItems) { if (!leaveItems) {
setUnreadCount(0); setUnreadCount(0);
last = 0; lastMsg = 0;
} }
height = 0; height = 0;
oldLoaded = false; oldLoaded = false;
@ -1478,7 +1514,7 @@ void HistoryItem::destroy() {
bool wasAtBottom = history()->loadedAtBottom(); bool wasAtBottom = history()->loadedAtBottom();
_history->removeNotification(this); _history->removeNotification(this);
detach(); detach();
if (history()->last == this) { if (history()->lastMsg == this) {
history()->fixLastMessage(wasAtBottom); history()->fixLastMessage(wasAtBottom);
} }
HistoryMedia *m = getMedia(true); HistoryMedia *m = getMedia(true);
@ -1554,8 +1590,7 @@ HistoryPhoto::HistoryPhoto(const MTPDphoto &photo, const QString &caption, Histo
, _caption(st::minPhotoSize) , _caption(st::minPhotoSize)
, openl(new PhotoLink(data)) { , openl(new PhotoLink(data)) {
if (!caption.isEmpty()) { if (!caption.isEmpty()) {
bool bot = (!parent->history()->peer->chat && parent->history()->peer->asUser()->botInfo) || (!parent->from()->chat && parent->from()->asUser()->botInfo); _caption.setText(st::msgFont, caption + textcmdSkipBlock(parent->timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), itemTextParseOptions(parent));
_caption.setText(st::msgFont, caption + textcmdSkipBlock(parent->timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), bot ? _historyBotOptions : _historyTextOptions);
} }
init(); init();
} }
@ -1955,8 +1990,7 @@ HistoryVideo::HistoryVideo(const MTPDvideo &video, const QString &caption, Histo
, _uplDone(0) , _uplDone(0)
{ {
if (!caption.isEmpty()) { if (!caption.isEmpty()) {
bool bot = (!parent->history()->peer->chat && parent->history()->peer->asUser()->botInfo) || (!parent->from()->chat && parent->from()->asUser()->botInfo); _caption.setText(st::msgFont, caption + textcmdSkipBlock(parent->timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), itemTextParseOptions(parent));
_caption.setText(st::msgFont, caption + textcmdSkipBlock(parent->timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), bot ? _historyBotOptions : _historyTextOptions);
} }
_size = formatDurationAndSizeText(data->duration, data->size); _size = formatDurationAndSizeText(data->duration, data->size);
@ -4664,11 +4698,10 @@ void HistoryMessage::initMediaFromDocument(DocumentData *doc) {
void HistoryMessage::initDimensions(const QString &text) { void HistoryMessage::initDimensions(const QString &text) {
if (!_media || !text.isEmpty()) { // !justMedia() if (!_media || !text.isEmpty()) { // !justMedia()
bool bot = (!history()->peer->chat && history()->peer->asUser()->botInfo) || (!from()->chat && from()->asUser()->botInfo);
if (_media && _media->isDisplayed()) { if (_media && _media->isDisplayed()) {
_text.setText(st::msgFont, text, bot ? _historyBotOptions : _historyTextOptions); _text.setText(st::msgFont, text, itemTextParseOptions(this));
} else { } else {
_text.setText(st::msgFont, text + textcmdSkipBlock(timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), bot ? _historyBotOptions : _historyTextOptions); _text.setText(st::msgFont, text + textcmdSkipBlock(timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), itemTextParseOptions(this));
} }
} }
} }
@ -4684,18 +4717,17 @@ void HistoryMessage::initDimensions(const HistoryItem *parent) {
_maxw += st::msgPadding.left() + st::msgPadding.right(); _maxw += st::msgPadding.left() + st::msgPadding.right();
if (_media) { if (_media) {
_media->initDimensions(this); _media->initDimensions(this);
bool bot = (!history()->peer->chat && history()->peer->asUser()->botInfo) || (!from()->chat && from()->asUser()->botInfo);
if (_media->isDisplayed() && _text.hasSkipBlock()) { if (_media->isDisplayed() && _text.hasSkipBlock()) {
QString was = HistoryMessage::selectedText(FullItemSel); QString was = HistoryMessage::selectedText(FullItemSel);
if (!was.isEmpty()) { if (!was.isEmpty()) {
_text.setText(st::msgFont, was, bot ? _historyBotOptions : _historyTextOptions); // without date skip _text.setText(st::msgFont, was, itemTextParseOptions(this)); // without date skip
_textWidth = 0; _textWidth = 0;
_textHeight = 0; _textHeight = 0;
} }
} else if (!_media->isDisplayed() && !_text.hasSkipBlock()) { } else if (!_media->isDisplayed() && !_text.hasSkipBlock()) {
QString was = HistoryMessage::selectedText(FullItemSel); QString was = HistoryMessage::selectedText(FullItemSel);
if (!was.isEmpty()) { if (!was.isEmpty()) {
_text.setText(st::msgFont, was + textcmdSkipBlock(timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), bot ? _historyBotOptions : _historyTextOptions); // without date skip _text.setText(st::msgFont, was + textcmdSkipBlock(timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), itemTextParseOptions(this)); // without date skip
_textWidth = 0; _textWidth = 0;
_textHeight = 0; _textHeight = 0;
} }
@ -4749,18 +4781,17 @@ void HistoryMessage::setMedia(const MTPmessageMedia &media) {
} }
QString t; QString t;
initMedia(media, t); initMedia(media, t);
bool bot = (!history()->peer->chat && history()->peer->asUser()->botInfo) || (!from()->chat && from()->asUser()->botInfo);
if (_media && _media->isDisplayed() && !mediaWasDisplayed) { if (_media && _media->isDisplayed() && !mediaWasDisplayed) {
QString was = HistoryMessage::selectedText(FullItemSel); QString was = HistoryMessage::selectedText(FullItemSel);
if (!was.isEmpty()) { if (!was.isEmpty()) {
_text.setText(st::msgFont, was, bot ? _historyBotOptions : _historyTextOptions); // without date skip _text.setText(st::msgFont, was, itemTextParseOptions(this)); // without date skip
_textWidth = 0; _textWidth = 0;
_textHeight = 0; _textHeight = 0;
} }
} else if (mediaWasDisplayed && (!_media || !_media->isDisplayed())) { } else if (mediaWasDisplayed && (!_media || !_media->isDisplayed())) {
QString was = HistoryMessage::selectedText(FullItemSel); QString was = HistoryMessage::selectedText(FullItemSel);
if (!was.isEmpty()) { if (!was.isEmpty()) {
_text.setText(st::msgFont, was + textcmdSkipBlock(timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), bot ? _historyBotOptions : _historyTextOptions); // without date skip _text.setText(st::msgFont, was + textcmdSkipBlock(timeWidth(true), st::msgDateFont->height - st::msgDateDelta.y()), itemTextParseOptions(this)); // without date skip
_textWidth = 0; _textWidth = 0;
_textHeight = 0; _textHeight = 0;
} }
@ -5069,6 +5100,9 @@ HistoryMessage::~HistoryMessage() {
_media->unregItem(this); _media->unregItem(this);
delete _media; delete _media;
} }
if (_flags & MTPDmessage::flag_reply_markup) {
App::clearReplyMarkup(id);
}
} }
HistoryForwarded::HistoryForwarded(History *history, HistoryBlock *block, const MTPDmessage &msg) : HistoryMessage(history, block, msg.vid.v, msg.vflags.v, ::date(msg.vdate), msg.vfrom_id.v, textClean(qs(msg.vmessage)), msg.vmedia) HistoryForwarded::HistoryForwarded(History *history, HistoryBlock *block, const MTPDmessage &msg) : HistoryMessage(history, block, msg.vid.v, msg.vflags.v, ::date(msg.vdate), msg.vfrom_id.v, textClean(qs(msg.vmessage)), msg.vmedia)

View File

@ -198,7 +198,7 @@ struct History : public QList<HistoryBlock*> {
PeerData *peer; PeerData *peer;
bool oldLoaded, newLoaded; bool oldLoaded, newLoaded;
HistoryItem *last; HistoryItem *lastMsg;
MsgId activeMsgId; MsgId activeMsgId;
typedef QList<HistoryItem*> NotifyQueue; typedef QList<HistoryItem*> NotifyQueue;
@ -238,8 +238,8 @@ struct History : public QList<HistoryBlock*> {
} }
} }
} }
if (last == old) { if (lastMsg == old) {
last = item; lastMsg = item;
} }
// showFrom can't be detached // showFrom can't be detached
} }
@ -251,6 +251,9 @@ struct History : public QList<HistoryBlock*> {
int32 lastWidth, lastScrollTop; int32 lastWidth, lastScrollTop;
bool mute; bool mute;
MsgId lastKeyboardId;
PeerId lastKeyboardFrom;
mtpRequestId sendRequestId; mtpRequestId sendRequestId;
// for dialog drawing // for dialog drawing
@ -284,8 +287,14 @@ struct History : public QList<HistoryBlock*> {
static const int32 ScrollMax = INT_MAX; static const int32 ScrollMax = INT_MAX;
}; };
enum DialogsSortMode {
DialogsSortByDate,
DialogsSortByName,
DialogsSortByAdd
};
struct DialogsList { struct DialogsList {
DialogsList(bool sortByName) : begin(&last), end(&last), byName(sortByName), count(0), current(&last) { DialogsList(DialogsSortMode sortMode) : begin(&last), end(&last), sortMode(sortMode), count(0), current(&last) {
} }
void adjustCurrent(int32 y, int32 h) const { void adjustCurrent(int32 y, int32 h) const {
@ -323,10 +332,10 @@ struct DialogsList {
end->pos++; end->pos++;
if (begin == end) { if (begin == end) {
begin = current = result; begin = current = result;
if (!byName && updatePos) history->posInDialogs = 0; if (sortMode == DialogsSortByDate && updatePos) history->posInDialogs = 0;
} else { } else {
end->prev->next = result; end->prev->next = result;
if (!byName && updatePos) history->posInDialogs = end->prev->history->posInDialogs + 1; if (sortMode == DialogsSortByDate && updatePos) history->posInDialogs = end->prev->history->posInDialogs + 1;
} }
rowByPeer.insert(history->peer->id, result); rowByPeer.insert(history->peer->id, result);
++count; ++count;
@ -334,7 +343,7 @@ struct DialogsList {
} }
void bringToTop(DialogRow *row, bool updatePos = true) { void bringToTop(DialogRow *row, bool updatePos = true) {
if (!byName && updatePos && row != begin) { if (sortMode == DialogsSortByDate && updatePos && row != begin) {
row->history->posInDialogs = begin->history->posInDialogs - 1; row->history->posInDialogs = begin->history->posInDialogs - 1;
} }
insertBefore(row, begin); insertBefore(row, begin);
@ -389,7 +398,7 @@ struct DialogsList {
} }
DialogRow *adjustByName(const PeerData *peer) { DialogRow *adjustByName(const PeerData *peer) {
if (!byName) return 0; if (sortMode != DialogsSortByName) return 0;
RowByPeer::iterator i = rowByPeer.find(peer->id); RowByPeer::iterator i = rowByPeer.find(peer->id);
if (i == rowByPeer.cend()) return 0; if (i == rowByPeer.cend()) return 0;
@ -408,7 +417,7 @@ struct DialogsList {
} }
DialogRow *addByName(History *history) { DialogRow *addByName(History *history) {
if (!byName) return 0; if (sortMode != DialogsSortByName) return 0;
DialogRow *row = addToEnd(history), *change = row; DialogRow *row = addToEnd(history), *change = row;
const QString &peerName(history->peer->name); const QString &peerName(history->peer->name);
@ -425,7 +434,7 @@ struct DialogsList {
} }
void adjustByPos(DialogRow *row) { void adjustByPos(DialogRow *row) {
if (byName) return; if (sortMode != DialogsSortByDate) return;
DialogRow *change = row; DialogRow *change = row;
while (change->prev && change->prev->history->posInDialogs > row->history->posInDialogs) { while (change->prev && change->prev->history->posInDialogs > row->history->posInDialogs) {
@ -440,7 +449,7 @@ struct DialogsList {
} }
DialogRow *addByPos(History *history) { DialogRow *addByPos(History *history) {
if (byName) return 0; if (sortMode != DialogsSortByDate) return 0;
DialogRow *row = addToEnd(history, false); DialogRow *row = addToEnd(history, false);
adjustByPos(row); adjustByPos(row);
@ -475,7 +484,7 @@ struct DialogsList {
DialogRow last; DialogRow last;
DialogRow *begin, *end; DialogRow *begin, *end;
bool byName; DialogsSortMode sortMode;
int32 count; int32 count;
typedef QHash<PeerId, DialogRow*> RowByPeer; typedef QHash<PeerId, DialogRow*> RowByPeer;
@ -485,7 +494,7 @@ struct DialogsList {
}; };
struct DialogsIndexed { struct DialogsIndexed {
DialogsIndexed(bool sortByName) : byName(sortByName), list(byName) { DialogsIndexed(DialogsSortMode sortMode) : sortMode(sortMode), list(sortMode) {
} }
History::DialogLinks addToEnd(History *history) { History::DialogLinks addToEnd(History *history) {
@ -499,7 +508,7 @@ struct DialogsIndexed {
for (PeerData::NameFirstChars::const_iterator i = history->peer->chars.cbegin(), e = history->peer->chars.cend(); i != e; ++i) { for (PeerData::NameFirstChars::const_iterator i = history->peer->chars.cbegin(), e = history->peer->chars.cend(); i != e; ++i) {
DialogsIndex::iterator j = index.find(*i); DialogsIndex::iterator j = index.find(*i);
if (j == index.cend()) { if (j == index.cend()) {
j = index.insert(*i, new DialogsList(byName)); j = index.insert(*i, new DialogsList(sortMode));
} }
result.insert(*i, j.value()->addToEnd(history)); result.insert(*i, j.value()->addToEnd(history));
} }
@ -517,7 +526,7 @@ struct DialogsIndexed {
for (PeerData::NameFirstChars::const_iterator i = history->peer->chars.cbegin(), e = history->peer->chars.cend(); i != e; ++i) { for (PeerData::NameFirstChars::const_iterator i = history->peer->chars.cbegin(), e = history->peer->chars.cend(); i != e; ++i) {
DialogsIndex::iterator j = index.find(*i); DialogsIndex::iterator j = index.find(*i);
if (j == index.cend()) { if (j == index.cend()) {
j = index.insert(*i, new DialogsList(byName)); j = index.insert(*i, new DialogsList(sortMode));
} }
j.value()->addByName(history); j.value()->addByName(history);
} }
@ -556,7 +565,7 @@ struct DialogsIndexed {
void clear(); void clear();
bool byName; DialogsSortMode sortMode;
DialogsList list; DialogsList list;
typedef QMap<QChar, DialogsList*> DialogsIndex; typedef QMap<QChar, DialogsList*> DialogsIndex;
DialogsIndex index; DialogsIndex index;
@ -678,6 +687,9 @@ public:
void markMediaRead() { void markMediaRead() {
_flags &= ~MTPDmessage_flag_media_unread; _flags &= ~MTPDmessage_flag_media_unread;
} }
bool hasReplyMarkup() const {
return _flags & MTPDmessage::flag_reply_markup;
}
virtual bool needCheck() const { virtual bool needCheck() const {
return true; return true;
} }

File diff suppressed because it is too large Load Diff

View File

@ -180,6 +180,7 @@ public:
void insertFromMimeData(const QMimeData *source); void insertFromMimeData(const QMimeData *source);
void focusInEvent(QFocusEvent *e); void focusInEvent(QFocusEvent *e);
void setMaxHeight(int32 maxHeight);
public slots: public slots:
@ -193,6 +194,67 @@ signals:
private: private:
HistoryWidget *history; HistoryWidget *history;
int32 _maxHeight;
};
class BotKeyboard : public QWidget {
Q_OBJECT
public:
BotKeyboard();
void paintEvent(QPaintEvent *e);
void resizeEvent(QResizeEvent *e);
void mousePressEvent(QMouseEvent *e);
void mouseMoveEvent(QMouseEvent *e);
void mouseReleaseEvent(QMouseEvent *e);
void leaveEvent(QEvent *e);
bool updateMarkup(HistoryItem *last);
bool hasMarkup() const;
bool hoverStep(float64 ms);
void resizeToWidth(int32 width);
MsgId forMsgId() const {
return _wasForMsgId;
}
public slots:
void showCommandTip();
void updateSelected();
private:
void updateStyle(int32 w = -1);
void clearSelection();
MsgId _wasForMsgId;
QTimer _cmdTipTimer;
QPoint _lastMousePos;
struct Button {
Button(const QString &str = QString()) : cmd(str), text(1), cwidth(0), hover(0), full(true) {
}
QRect rect;
QString cmd;
Text text;
int32 cwidth;
float64 hover;
bool full;
};
int32 _sel, _down;
QList<QList<Button> > _btns;
typedef QMap<int32, uint64> Animations;
Animations _animations;
Animation _hoverAnim;
const style::botKeyboardButton *_st;
}; };
class HistoryHider : public QWidget, public Animated { class HistoryHider : public QWidget, public Animated {
@ -379,6 +441,8 @@ public:
bool recordingStep(float64 ms); bool recordingStep(float64 ms);
void stopRecording(bool send); void stopRecording(bool send);
void sendBotCommand(const QString &cmd, MsgId replyTo);
~HistoryWidget(); ~HistoryWidget();
signals: signals:
@ -399,7 +463,7 @@ public slots:
void onPreviewTimeout(); void onPreviewTimeout();
void peerUpdated(PeerData *data); void peerUpdated(PeerData *data);
void onPeerLoaded(PeerData *data); void onFullPeerUpdated(PeerData *data);
void cancelTyping(); void cancelTyping();
@ -423,6 +487,8 @@ public slots:
void onPhotoDrop(QDropEvent *e); void onPhotoDrop(QDropEvent *e);
void onDocumentDrop(QDropEvent *e); void onDocumentDrop(QDropEvent *e);
void onKbToggle();
void onPhotoReady(); void onPhotoReady();
void onSendConfirmed(); void onSendConfirmed();
void onSendCancelled(); void onSendCancelled();
@ -470,7 +536,11 @@ private:
int32 _replyToNameVersion; int32 _replyToNameVersion;
IconedButton _replyForwardPreviewCancel; IconedButton _replyForwardPreviewCancel;
void updateReplyToName(); void updateReplyToName();
void drawFieldBackground(QPainter &p);
void drawField(Painter &p);
void drawRecordButton(Painter &p);
void drawRecording(Painter &p);
void updateField();
QString _previewLinks; QString _previewLinks;
WebPageData *_previewData; WebPageData *_previewData;
@ -492,6 +562,8 @@ private:
void addMessagesToFront(const QVector<MTPMessage> &messages); void addMessagesToFront(const QVector<MTPMessage> &messages);
void addMessagesToBack(const QVector<MTPMessage> &messages); void addMessagesToBack(const QVector<MTPMessage> &messages);
void updateBotKeyboard();
void stickersGot(const MTPmessages_AllStickers &stickers); void stickersGot(const MTPmessages_AllStickers &stickers);
bool stickersFailed(const RPCError &error); bool stickersFailed(const RPCError &error);
@ -519,14 +591,14 @@ private:
ScrollArea _scroll; ScrollArea _scroll;
HistoryList *_list; HistoryList *_list;
History *hist; History *hist;
bool _histInited; // initial updateListSize() called bool _histInited, _histNeedUpdate; // initial updateListSize() called
IconedButton _toHistoryEnd; IconedButton _toHistoryEnd;
MentionsDropdown _attachMention; MentionsDropdown _attachMention;
FlatButton _send; FlatButton _send;
IconedButton _attachDocument, _attachPhoto, _attachEmoji; IconedButton _attachDocument, _attachPhoto, _attachEmoji, _kbShow, _kbHide;
MessageField _field; MessageField _field;
Animation _recordAnim, _recordingAnim; Animation _recordAnim, _recordingAnim;
bool _recording, _inRecord, _inField; bool _recording, _inRecord, _inField;
@ -536,6 +608,11 @@ private:
anim::cvalue a_recordCancel; anim::cvalue a_recordCancel;
int32 _recordCancelWidth; int32 _recordCancelWidth;
bool _kbShown, _kbWasHidden;
HistoryItem *_kbReplyTo;
ScrollArea _kbScroll;
BotKeyboard _keyboard;
Dropdown _attachType; Dropdown _attachType;
EmojiPan _emojiPan; EmojiPan _emojiPan;
DragState _attachDrag; DragState _attachDrag;

View File

@ -766,19 +766,21 @@ void MainWidget::removeContact(UserData *user) {
void MainWidget::addParticipants(ChatData *chat, const QVector<UserData*> &users) { void MainWidget::addParticipants(ChatData *chat, const QVector<UserData*> &users) {
for (QVector<UserData*>::const_iterator i = users.cbegin(), e = users.cend(); i != e; ++i) { for (QVector<UserData*>::const_iterator i = users.cbegin(), e = users.cend(); i != e; ++i) {
MTP::send(MTPmessages_AddChatUser(MTP_int(chat->id & 0xFFFFFFFF), (*i)->inputUser, MTP_int(ForwardOnAdd)), rpcDone(&MainWidget::sentUpdatesReceived), rpcFail(&MainWidget::addParticipantFail, chat), 0, 5); MTP::send(MTPmessages_AddChatUser(MTP_int(chat->id & 0xFFFFFFFF), (*i)->inputUser, MTP_int(ForwardOnAdd)), rpcDone(&MainWidget::sentUpdatesReceived), rpcFail(&MainWidget::addParticipantFail, *i), 0, 5);
} }
App::wnd()->hideLayer(); App::wnd()->hideLayer();
showPeer(chat->id, 0, false); showPeer(chat->id, 0, false);
} }
bool MainWidget::addParticipantFail(ChatData *chat, const RPCError &error) { bool MainWidget::addParticipantFail(UserData *user, const RPCError &error) {
if (error.type().startsWith(qsl("FLOOD_WAIT_"))) return false; if (error.type().startsWith(qsl("FLOOD_WAIT_"))) return false;
ConfirmBox *box = new ConfirmBox(lang(lng_failed_add_participant), true); QString text = lang(lng_failed_add_participant);
App::wnd()->showLayer(box);
if (error.type() == "USER_LEFT_CHAT") { // trying to return banned user to his group if (error.type() == "USER_LEFT_CHAT") { // trying to return banned user to his group
} else if (error.type() == "USER_ALREADY_PARTICIPANT" && user->botInfo) {
text = lang(lng_bot_already_in_group);
} }
App::wnd()->showLayer(new ConfirmBox(text, true));
return false; return false;
} }
@ -821,7 +823,7 @@ void MainWidget::checkedHistory(PeerData *peer, const MTPmessages_Messages &resu
dialogs.removePeer(peer); dialogs.removePeer(peer);
} else { } else {
History *h = App::historyLoaded(peer->id); History *h = App::historyLoaded(peer->id);
if (!h->last) { if (!h->lastMsg) {
h->addToBack((*v)[0], 0); h->addToBack((*v)[0], 0);
} }
} }
@ -933,6 +935,10 @@ DialogsIndexed &MainWidget::contactsList() {
return dialogs.contactsList(); return dialogs.contactsList();
} }
DialogsIndexed &MainWidget::dialogsList() {
return dialogs.dialogsList();
}
void MainWidget::sendPreparedText(History *hist, const QString &text, MsgId replyTo, WebPageId webPageId) { void MainWidget::sendPreparedText(History *hist, const QString &text, MsgId replyTo, WebPageId webPageId) {
saveRecentHashtags(text); saveRecentHashtags(text);
QString sendingText, leftText = text; QString sendingText, leftText = text;
@ -1016,10 +1022,8 @@ void MainWidget::stopAnimActive() {
history.stopAnimActive(); history.stopAnimActive();
} }
void MainWidget::sendBotCommand(const QString &cmd) { void MainWidget::sendBotCommand(const QString &cmd, MsgId replyTo) {
if (history.peer()) { history.sendBotCommand(cmd, replyTo);
sendMessage(App::history(history.peer()->id), cmd, 0);
}
} }
void MainWidget::searchMessages(const QString &query) { void MainWidget::searchMessages(const QString &query) {
@ -3229,7 +3233,7 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
case mtpc_updateChatParticipants: { case mtpc_updateChatParticipants: {
const MTPDupdateChatParticipants &d(update.c_updateChatParticipants()); const MTPDupdateChatParticipants &d(update.c_updateChatParticipants());
App::feedParticipants(d.vparticipants); App::feedParticipants(d.vparticipants, true);
} break; } break;
case mtpc_updateChatParticipantAdd: { case mtpc_updateChatParticipantAdd: {

View File

@ -270,7 +270,7 @@ public:
void removeContact(UserData *user); void removeContact(UserData *user);
void addParticipants(ChatData *chat, const QVector<UserData*> &users); void addParticipants(ChatData *chat, const QVector<UserData*> &users);
bool addParticipantFail(ChatData *chat, const RPCError &e); bool addParticipantFail(UserData *user, const RPCError &e);
void kickParticipant(ChatData *chat, UserData *user); void kickParticipant(ChatData *chat, UserData *user);
bool kickParticipantFail(ChatData *chat, const RPCError &e); bool kickParticipantFail(ChatData *chat, const RPCError &e);
@ -285,6 +285,7 @@ public:
void clearSelectedItems(); void clearSelectedItems();
DialogsIndexed &contactsList(); DialogsIndexed &contactsList();
DialogsIndexed &dialogsList();
void sendMessage(History *history, const QString &text, MsgId replyTo); void sendMessage(History *history, const QString &text, MsgId replyTo);
void sendPreparedText(History *hist, const QString &text, MsgId replyTo, WebPageId webPageId = 0); void sendPreparedText(History *hist, const QString &text, MsgId replyTo, WebPageId webPageId = 0);
@ -295,7 +296,7 @@ public:
uint64 animActiveTime() const; uint64 animActiveTime() const;
void stopAnimActive(); void stopAnimActive();
void sendBotCommand(const QString &cmd); void sendBotCommand(const QString &cmd, MsgId msgId);
void searchMessages(const QString &query); void searchMessages(const QString &query);
void preloadOverviews(PeerData *peer); void preloadOverviews(PeerData *peer);

View File

@ -39,9 +39,16 @@ ProfileInner::ProfileInner(ProfileWidget *profile, ScrollArea *scroll, const Pee
_addParticipant(this, lang(lng_profile_add_participant), st::btnShareContact), _addParticipant(this, lang(lng_profile_add_participant), st::btnShareContact),
_sendMessage(this, lang(lng_profile_send_message), st::btnShareContact), _sendMessage(this, lang(lng_profile_send_message), st::btnShareContact),
_shareContact(this, lang(lng_profile_share_contact), st::btnShareContact), _shareContact(this, lang(lng_profile_share_contact), st::btnShareContact),
_inviteToGroup(this, lang(lng_profile_invite_to_group), st::btnShareContact),
_cancelPhoto(this, lang(lng_cancel)), _cancelPhoto(this, lang(lng_cancel)),
_createInvitationLink(this, lang(lng_group_invite_create)), _createInvitationLink(this, lang(lng_group_invite_create)),
_invitationLink(this, qsl("telegram.me/joinchat/")), _invitationLink(this, qsl("telegram.me/joinchat/")),
_botSettings(this, lang(lng_profile_bot_settings)),
_botHelp(this, lang(lng_profile_bot_help)),
// about
_about(st::wndMinWidth - st::profilePadding.left() - st::profilePadding.right()),
_aboutTop(0), _aboutHeight(0),
a_photo(0), a_photo(0),
_photoOver(false), _photoOver(false),
@ -66,7 +73,7 @@ ProfileInner::ProfileInner(ProfileWidget *profile, ScrollArea *scroll, const Pee
_menu(0) { _menu(0) {
connect(App::api(), SIGNAL(fullPeerLoaded(PeerData*)), this, SLOT(onFullPeerLoaded(PeerData*))); connect(App::api(), SIGNAL(fullPeerUpdated(PeerData*)), this, SLOT(onFullPeerUpdated(PeerData*)));
if (_peerUser) { if (_peerUser) {
_phoneText = _peerUser->phone.isEmpty() ? QString() : App::formatPhone(_peerUser->phone); _phoneText = _peerUser->phone.isEmpty() ? QString() : App::formatPhone(_peerUser->phone);
@ -87,12 +94,16 @@ ProfileInner::ProfileInner(ProfileWidget *profile, ScrollArea *scroll, const Pee
connect(&_addParticipant, SIGNAL(clicked()), this, SLOT(onAddParticipant())); connect(&_addParticipant, SIGNAL(clicked()), this, SLOT(onAddParticipant()));
connect(&_sendMessage, SIGNAL(clicked()), this, SLOT(onSendMessage())); connect(&_sendMessage, SIGNAL(clicked()), this, SLOT(onSendMessage()));
connect(&_shareContact, SIGNAL(clicked()), this, SLOT(onShareContact())); connect(&_shareContact, SIGNAL(clicked()), this, SLOT(onShareContact()));
connect(&_inviteToGroup, SIGNAL(clicked()), this, SLOT(onInviteToGroup()));
connect(&_cancelPhoto, SIGNAL(clicked()), this, SLOT(onUpdatePhotoCancel())); connect(&_cancelPhoto, SIGNAL(clicked()), this, SLOT(onUpdatePhotoCancel()));
connect(&_createInvitationLink, SIGNAL(clicked()), this, SLOT(onCreateInvitationLink())); connect(&_createInvitationLink, SIGNAL(clicked()), this, SLOT(onCreateInvitationLink()));
connect(&_invitationLink, SIGNAL(clicked()), this, SLOT(onInvitationLink())); connect(&_invitationLink, SIGNAL(clicked()), this, SLOT(onInvitationLink()));
_invitationLink.setAcceptBoth(true); _invitationLink.setAcceptBoth(true);
updateInvitationLink(); updateInvitationLink();
connect(&_botSettings, SIGNAL(clicked()), this, SLOT(onBotSettings()));
connect(&_botHelp, SIGNAL(clicked()), this, SLOT(onBotHelp()));
connect(App::app(), SIGNAL(peerPhotoDone(PeerId)), this, SLOT(onPhotoUpdateDone(PeerId))); connect(App::app(), SIGNAL(peerPhotoDone(PeerId)), this, SLOT(onPhotoUpdateDone(PeerId)));
connect(App::app(), SIGNAL(peerPhotoFail(PeerId)), this, SLOT(onPhotoUpdateFail(PeerId))); connect(App::app(), SIGNAL(peerPhotoFail(PeerId)), this, SLOT(onPhotoUpdateFail(PeerId)));
@ -100,6 +111,17 @@ ProfileInner::ProfileInner(ProfileWidget *profile, ScrollArea *scroll, const Pee
connect(App::main(), SIGNAL(peerUpdated(PeerData *)), this, SLOT(peerUpdated(PeerData *))); connect(App::main(), SIGNAL(peerUpdated(PeerData *)), this, SLOT(peerUpdated(PeerData *)));
connect(App::main(), SIGNAL(peerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)), this, SLOT(peerUpdated(PeerData *))); connect(App::main(), SIGNAL(peerNameChanged(PeerData *, const PeerData::Names &, const PeerData::NameFirstChars &)), this, SLOT(peerUpdated(PeerData *)));
// about
if (_peerUser && _peerUser->botInfo) {
if (!_peerUser->botInfo->shareText.isEmpty()) {
_about.setText(st::linkFont, _peerUser->botInfo->shareText, _historyBotOptions);
}
updateBotLinksVisibility();
} else {
_botSettings.hide();
_botHelp.hide();
}
// settings // settings
connect(&_enableNotifications, SIGNAL(clicked()), this, SLOT(onEnableNotifications())); connect(&_enableNotifications, SIGNAL(clicked()), this, SLOT(onEnableNotifications()));
connect(&_clearHistory, SIGNAL(clicked()), this, SLOT(onClearHistory())); connect(&_clearHistory, SIGNAL(clicked()), this, SLOT(onClearHistory()));
@ -126,6 +148,10 @@ void ProfileInner::onShareContact() {
App::main()->shareContactLayer(_peerUser); App::main()->shareContactLayer(_peerUser);
} }
void ProfileInner::onInviteToGroup() {
App::wnd()->showLayer(new ContactsBox(_peerUser));
}
void ProfileInner::onSendMessage() { void ProfileInner::onSendMessage() {
App::main()->showPeer(_peer->id); App::main()->showPeer(_peer->id);
} }
@ -279,7 +305,7 @@ void ProfileInner::chatInviteDone(const MTPExportedChatInvite &result) {
App::wnd()->hideLayer(); App::wnd()->hideLayer();
} }
void ProfileInner::onFullPeerLoaded(PeerData *peer) { void ProfileInner::onFullPeerUpdated(PeerData *peer) {
if (peer != _peer) return; if (peer != _peer) return;
if (_peerUser) { if (_peerUser) {
PhotoData *userPhoto = _peerUser->photoId ? App::photo(_peerUser->photoId) : 0; PhotoData *userPhoto = _peerUser->photoId ? App::photo(_peerUser->photoId) : 0;
@ -288,6 +314,15 @@ void ProfileInner::onFullPeerLoaded(PeerData *peer) {
} else { } else {
_photoLink = TextLinkPtr(); _photoLink = TextLinkPtr();
} }
if (_peerUser->botInfo) {
if (_peerUser->botInfo->shareText.isEmpty()) {
_about = Text(st::wndMinWidth - st::profilePadding.left() - st::profilePadding.right());
} else {
_about.setText(st::linkFont, _peerUser->botInfo->shareText, _historyBotOptions);
}
updateBotLinksVisibility();
resizeEvent(0);
}
} else if (_peerChat) { } else if (_peerChat) {
updateInvitationLink(); updateInvitationLink();
showAll(); showAll();
@ -295,6 +330,30 @@ void ProfileInner::onFullPeerLoaded(PeerData *peer) {
} }
} }
void ProfileInner::onBotSettings() {
for (int32 i = 0, l = _peerUser->botInfo->commands.size(); i != l; ++i) {
QString cmd = _peerUser->botInfo->commands.at(i).command;
if (!cmd.compare(qsl("settings"), Qt::CaseInsensitive)) {
App::main()->showPeer(_peer->id);
App::main()->sendBotCommand('/' + cmd, 0);
return;
}
}
updateBotLinksVisibility();
}
void ProfileInner::onBotHelp() {
for (int32 i = 0, l = _peerUser->botInfo->commands.size(); i != l; ++i) {
QString cmd = _peerUser->botInfo->commands.at(i).command;
if (!cmd.compare(qsl("help"), Qt::CaseInsensitive)) {
App::main()->showPeer(_peer->id);
App::main()->sendBotCommand('/' + cmd, 0);
return;
}
}
updateBotLinksVisibility();
}
void ProfileInner::peerUpdated(PeerData *data) { void ProfileInner::peerUpdated(PeerData *data) {
if (data == _peer) { if (data == _peer) {
PhotoData *photo = 0; PhotoData *photo = 0;
@ -446,7 +505,7 @@ void ProfileInner::paintEvent(QPaintEvent *e) {
if (!_errorText.isEmpty()) { if (!_errorText.isEmpty()) {
p.setFont(st::setErrFont->f); p.setFont(st::setErrFont->f);
p.setPen(st::setErrColor->p); p.setPen(st::setErrColor->p);
p.drawText(_left + st::profilePhotoSize + st::profilePhoneLeft, top + addbyname + st::profilePhoneTop + st::profilePhoneFont->ascent, _errorText); p.drawText(_left + st::profilePhotoSize + st::profilePhoneLeft, _cancelPhoto.y() + addbyname + st::profilePhoneFont->ascent, _errorText);
} }
if (!_phoneText.isEmpty()) { if (!_phoneText.isEmpty()) {
p.setPen(st::black->p); p.setPen(st::black->p);
@ -464,6 +523,17 @@ void ProfileInner::paintEvent(QPaintEvent *e) {
} }
top += _shareContact.height(); top += _shareContact.height();
// about
if (!_about.isEmpty()) {
p.setFont(st::profileHeaderFont->f);
p.setPen(st::profileHeaderColor->p);
p.drawText(_left + st::profileHeaderLeft, top + st::profileHeaderTop + st::profileHeaderFont->ascent, lang(lng_profile_about_section));
top += st::profileHeaderSkip;
_about.draw(p, _left, top, _width);
top += _aboutHeight;
}
// settings // settings
p.setFont(st::profileHeaderFont->f); p.setFont(st::profileHeaderFont->f);
p.setPen(st::profileHeaderColor->p); p.setPen(st::profileHeaderColor->p);
@ -537,7 +607,15 @@ void ProfileInner::paintEvent(QPaintEvent *e) {
if (!data) { if (!data) {
data = _participantsData[cnt] = new ParticipantData(); data = _participantsData[cnt] = new ParticipantData();
data->name.setText(st::profileListNameFont, user->name, _textNameOptions); data->name.setText(st::profileListNameFont, user->name, _textNameOptions);
data->online = App::onlineText(user, l_time); if (user->botInfo) {
if (user->botInfo->readsAllHistory) {
data->online = lang(lng_status_bot_reads_all);
} else {
data->online = lang(lng_status_bot_not_reads_all);
}
} else {
data->online = App::onlineText(user, l_time);
}
data->cankick = (user != App::self()) && (_chatAdmin || (_peerChat->cankick.constFind(user) != _peerChat->cankick.cend())); data->cankick = (user != App::self()) && (_chatAdmin || (_peerChat->cankick.constFind(user) != _peerChat->cankick.cend()));
} }
p.setPen(st::profileListNameColor->p); p.setPen(st::profileListNameColor->p);
@ -579,9 +657,9 @@ void ProfileInner::mouseMoveEvent(QMouseEvent *e) {
} }
} }
if (!_photoLink && (!_peerChat || _peerChat->forbidden)) { if (!_photoLink && (!_peerChat || _peerChat->forbidden)) {
setCursor((_kickOver || _kickDown) ? style::cur_pointer : style::cur_default); setCursor((_kickOver || _kickDown || textlnkOver()) ? style::cur_pointer : style::cur_default);
} else { } else {
setCursor((_kickOver || _kickDown || _photoOver) ? style::cur_pointer : style::cur_default); setCursor((_kickOver || _kickDown || _photoOver || textlnkOver()) ? style::cur_pointer : style::cur_default);
} }
} }
@ -590,6 +668,16 @@ void ProfileInner::updateSelected() {
QPoint lp = mapFromGlobal(_lastPos); QPoint lp = mapFromGlobal(_lastPos);
TextLinkPtr lnk;
bool inText = false;
if (!_about.isEmpty() && lp.y() >= _aboutTop && lp.y() < _aboutTop + _aboutHeight && lp.x() >= _left && lp.x() < _left + _width) {
_about.getState(lnk, inText, lp.x() - _left, lp.y() - _aboutTop, _width);
}
if (textlnkOver() != lnk) {
textlnkOver(lnk);
update(QRect(_left, _aboutTop, _width, _aboutHeight));
}
int32 partfrom = _mediaAudios.y() + _mediaAudios.height() + st::profileHeaderSkip; int32 partfrom = _mediaAudios.y() + _mediaAudios.height() + st::profileHeaderSkip;
int32 newSelected = (lp.x() >= _left - st::profileListPadding.width() && lp.x() < _left + _width + st::profileListPadding.width() && lp.y() >= partfrom) ? (lp.y() - partfrom) / _pHeight : -1; int32 newSelected = (lp.x() >= _left - st::profileListPadding.width() && lp.x() < _left + _width + st::profileListPadding.width() && lp.y() >= partfrom) ? (lp.y() - partfrom) / _pHeight : -1;
@ -631,18 +719,31 @@ void ProfileInner::mousePressEvent(QMouseEvent *e) {
onUpdatePhoto(); onUpdatePhoto();
} }
} }
textlnkDown(textlnkOver());
} }
} }
void ProfileInner::mouseReleaseEvent(QMouseEvent *e) { void ProfileInner::mouseReleaseEvent(QMouseEvent *e) {
_lastPos = e->globalPos();
updateSelected();
if (_kickDown && _kickDown == _kickOver) { if (_kickDown && _kickDown == _kickOver) {
_kickConfirm = _kickOver; _kickConfirm = _kickOver;
ConfirmBox *box = new ConfirmBox(lng_profile_sure_kick(lt_user, _kickOver->firstName)); ConfirmBox *box = new ConfirmBox(lng_profile_sure_kick(lt_user, _kickOver->firstName));
connect(box, SIGNAL(confirmed()), this, SLOT(onKickConfirm())); connect(box, SIGNAL(confirmed()), this, SLOT(onKickConfirm()));
App::wnd()->showLayer(box); App::wnd()->showLayer(box);
} }
if (textlnkDown()) {
TextLinkPtr lnk = textlnkDown();
textlnkDown(TextLinkPtr());
if (lnk == textlnkOver()) {
if (reBotCommand().match(lnk->encoded()).hasMatch()) {
App::main()->showPeer(_peer->id);
}
lnk->onClick(e->button());
}
}
_kickDown = 0; _kickDown = 0;
setCursor(_kickOver ? style::cur_pointer : style::cur_default); setCursor((_kickOver || textlnkOver()) ? style::cur_pointer : style::cur_default);
update(); update();
} }
@ -697,6 +798,8 @@ void ProfileInner::resizeEvent(QResizeEvent *e) {
_cancelPhoto.move(_left + _width - _cancelPhoto.width(), top + st::profilePhotoSize - st::linkFont->height); _cancelPhoto.move(_left + _width - _cancelPhoto.width(), top + st::profilePhotoSize - st::linkFont->height);
} else { } else {
_cancelPhoto.move(_left + _width - _cancelPhoto.width(), top + st::profilePhoneTop); _cancelPhoto.move(_left + _width - _cancelPhoto.width(), top + st::profilePhoneTop);
_botSettings.move(_left + st::profilePhotoSize + st::profilePhoneLeft, top + st::profileStatusTop + st::linkFont->ascent - (st::profileNameTop + st::profileNameFont->ascent) + st::profilePhoneTop);
_botHelp.move(_botSettings.x() + (_botSettings.isHidden() ? 0 : _botSettings.width() + st::profilePhoneLeft), _botSettings.y());
} }
top += st::profilePhotoSize; top += st::profilePhotoSize;
@ -705,8 +808,17 @@ void ProfileInner::resizeEvent(QResizeEvent *e) {
_sendMessage.setGeometry(_left, top, btnWidth, _sendMessage.height()); _sendMessage.setGeometry(_left, top, btnWidth, _sendMessage.height());
_addParticipant.setGeometry(_left + _width - btnWidth, top, btnWidth, _addParticipant.height()); _addParticipant.setGeometry(_left + _width - btnWidth, top, btnWidth, _addParticipant.height());
_shareContact.setGeometry(_left + _width - btnWidth, top, btnWidth, _shareContact.height()); _shareContact.setGeometry(_left + _width - btnWidth, top, btnWidth, _shareContact.height());
_inviteToGroup.setGeometry(_left + _width - btnWidth, top, btnWidth, _inviteToGroup.height());
top += _shareContact.height(); top += _shareContact.height();
// about
if (!_about.isEmpty()) {
top += st::profileHeaderSkip;
_aboutTop = top; _aboutHeight = _about.countHeight(_width); top += _aboutHeight;
} else {
_aboutTop = _aboutHeight = 0;
}
// settings // settings
top += st::profileHeaderSkip; top += st::profileHeaderSkip;
_enableNotifications.move(_left, top); top += _enableNotifications.height(); _enableNotifications.move(_left, top); top += _enableNotifications.height();
@ -829,6 +941,7 @@ void ProfileInner::showAll() {
if (_peerChat) { if (_peerChat) {
_sendMessage.hide(); _sendMessage.hide();
_shareContact.hide(); _shareContact.hide();
_inviteToGroup.hide();
if (_peerChat->forbidden) { if (_peerChat->forbidden) {
_uploadPhoto.hide(); _uploadPhoto.hide();
_cancelPhoto.hide(); _cancelPhoto.hide();
@ -871,8 +984,14 @@ void ProfileInner::showAll() {
_sendMessage.show(); _sendMessage.show();
if (_peerUser->phone.isEmpty()) { if (_peerUser->phone.isEmpty()) {
_shareContact.hide(); _shareContact.hide();
if (_peerUser->botInfo && !_peerUser->botInfo->cantJoinGroups) {
_inviteToGroup.show();
} else {
_inviteToGroup.hide();
}
} else { } else {
_shareContact.show(); _shareContact.show();
_inviteToGroup.hide();
} }
_enableNotifications.show(); _enableNotifications.show();
_clearHistory.show(); _clearHistory.show();
@ -944,6 +1063,23 @@ void ProfileInner::updateInvitationLink() {
} }
} }
void ProfileInner::updateBotLinksVisibility() {
if (!_peerUser || !_peerUser->botInfo || _peerUser->botInfo->commands.isEmpty()) {
_botSettings.hide();
_botHelp.hide();
return;
}
bool hasSettings = false, hasHelp = false;
for (int32 i = 0, l = _peerUser->botInfo->commands.size(); i != l; ++i) {
QString cmd = _peerUser->botInfo->commands.at(i).command;
hasSettings |= !cmd.compare(qsl("settings"), Qt::CaseInsensitive);
hasHelp |= !cmd.compare(qsl("help"), Qt::CaseInsensitive);
if (hasSettings && hasHelp) break;
}
_botSettings.setVisible(hasSettings);
_botHelp.setVisible(hasHelp);
}
QString ProfileInner::overviewLinkText(int32 type, int32 count) { QString ProfileInner::overviewLinkText(int32 type, int32 count) {
switch (type) { switch (type) {
case OverviewPhotos: return lng_profile_photos(lt_count, count); case OverviewPhotos: return lng_profile_photos(lt_count, count);

View File

@ -64,6 +64,7 @@ public slots:
void deleteContextImage(); void deleteContextImage();
void onShareContact(); void onShareContact();
void onInviteToGroup();
void onSendMessage(); void onSendMessage();
void onEnableNotifications(); void onEnableNotifications();
@ -94,12 +95,16 @@ public slots:
void onCreateInvitationLink(); void onCreateInvitationLink();
void onCreateInvitationLinkSure(); void onCreateInvitationLinkSure();
void onFullPeerLoaded(PeerData *peer); void onFullPeerUpdated(PeerData *peer);
void onBotSettings();
void onBotHelp();
private: private:
void showAll(); void showAll();
void updateInvitationLink(); void updateInvitationLink();
void updateBotLinksVisibility();
void chatInviteDone(const MTPExportedChatInvite &result); void chatInviteDone(const MTPExportedChatInvite &result);
@ -120,9 +125,13 @@ private:
QString _phoneText; QString _phoneText;
TextLinkPtr _photoLink; TextLinkPtr _photoLink;
FlatButton _uploadPhoto, _addParticipant; FlatButton _uploadPhoto, _addParticipant;
FlatButton _sendMessage, _shareContact; FlatButton _sendMessage, _shareContact, _inviteToGroup;
LinkButton _cancelPhoto, _createInvitationLink, _invitationLink; LinkButton _cancelPhoto, _createInvitationLink, _invitationLink;
QString _invitationText; QString _invitationText;
LinkButton _botSettings, _botHelp;
Text _about;
int32 _aboutTop, _aboutHeight;
anim::fvalue a_photo; anim::fvalue a_photo;
bool _photoOver; bool _photoOver;

View File

@ -139,6 +139,7 @@ QUrl gUpdateURL = QUrl(qsl("http://tdesktop.com/linux/tupdates/current"));
#endif #endif
bool gContactsReceived = false; bool gContactsReceived = false;
bool gDialogsReceived = false;
bool gWideMode = true; bool gWideMode = true;

View File

@ -290,6 +290,7 @@ DeclareReadSetting(DBIPlatform, Platform);
DeclareReadSetting(QUrl, UpdateURL); DeclareReadSetting(QUrl, UpdateURL);
DeclareSetting(bool, ContactsReceived); DeclareSetting(bool, ContactsReceived);
DeclareSetting(bool, DialogsReceived);
DeclareSetting(bool, WideMode); DeclareSetting(bool, WideMode);

View File

@ -202,7 +202,10 @@ void UserData::setPhone(const QString &newPhone) {
} }
void UserData::setBotInfoVersion(int32 version) { void UserData::setBotInfoVersion(int32 version) {
if (!botInfo) { if (version < 0) {
delete botInfo;
botInfo = 0;
} else if (!botInfo) {
botInfo = new BotInfo(); botInfo = new BotInfo();
botInfo->version = version; botInfo->version = version;
} else if (botInfo->version < version) { } else if (botInfo->version < version) {
@ -229,10 +232,10 @@ void UserData::setBotInfo(const MTPBotInfo &info) {
setBotInfoVersion(d.vversion.v); setBotInfoVersion(d.vversion.v);
} }
QString desc = qs(d.vdescription) + "\n\nhttps://telegram.org test #test test /help test"; QString desc = qs(d.vdescription);
if (botInfo->description != desc) { if (botInfo->description != desc) {
botInfo->description = desc; botInfo->description = desc;
botInfo->text = Text(); botInfo->text = Text(st::msgMinWidth);
} }
botInfo->shareText = qs(d.vshare_text); botInfo->shareText = qs(d.vshare_text);

View File

@ -124,9 +124,10 @@ struct BotCommand {
QString command, params, description; QString command, params, description;
}; };
struct BotInfo { struct BotInfo {
BotInfo() : inited(false), version(0), text(st::msgMinWidth) { BotInfo() : inited(false), readsAllHistory(false), cantJoinGroups(false), version(0), text(st::msgMinWidth) {
} }
bool inited; bool inited;
bool readsAllHistory, cantJoinGroups;
int32 version; int32 version;
QString shareText, description; QString shareText, description;
QList<BotCommand> commands; QList<BotCommand> commands;
@ -164,7 +165,7 @@ struct UserData : public PeerData {
}; };
struct ChatData : public PeerData { struct ChatData : public PeerData {
ChatData(const PeerId &id) : PeerData(id), count(0), date(0), version(0), left(false), forbidden(true), photoId(0) { ChatData(const PeerId &id) : PeerData(id), count(0), date(0), version(0), left(false), forbidden(true), botStatus(0), photoId(0) {
} }
void setPhoto(const MTPChatPhoto &photo, const PhotoId &phId = 0); void setPhoto(const MTPChatPhoto &photo, const PhotoId &phId = 0);
int32 count; int32 count;
@ -179,6 +180,7 @@ struct ChatData : public PeerData {
CanKick cankick; CanKick cankick;
typedef QList<UserData*> LastAuthors; typedef QList<UserData*> LastAuthors;
LastAuthors lastAuthors; LastAuthors lastAuthors;
int32 botStatus; // -1 - no bots, 0 - unknown, 1 - one bot, that sees all history, 2 - other
ImagePtr photoFull; ImagePtr photoFull;
PhotoId photoId; PhotoId photoId;
QString invitationUrl; QString invitationUrl;

View File

@ -327,6 +327,8 @@ enum DBIScale {
dbisScaleCount = 5, dbisScaleCount = 5,
}; };
static const int MatrixRowShift = 40000;
enum DBIEmojiTab { enum DBIEmojiTab {
dbietRecent = -1, dbietRecent = -1,
dbietPeople = 0, dbietPeople = 0,
@ -339,7 +341,6 @@ enum DBIEmojiTab {
dbietStickers = 666, dbietStickers = 666,
}; };
static const int emojiTabCount = 8; static const int emojiTabCount = 8;
static const int emojiTabShift = 100000;
inline DBIEmojiTab emojiTabAtIndex(int index) { inline DBIEmojiTab emojiTabAtIndex(int index) {
return (index < 0 || index >= emojiTabCount) ? dbietRecent : DBIEmojiTab(index - 1); return (index < 0 || index >= emojiTabCount) ? dbietRecent : DBIEmojiTab(index - 1);
} }

View File

@ -557,6 +557,7 @@ void Window::checkAutoLock() {
void Window::setupIntro(bool anim) { void Window::setupIntro(bool anim) {
cSetContactsReceived(false); cSetContactsReceived(false);
cSetDialogsReceived(false);
if (intro && (intro->animating() || intro->isVisible()) && !main) return; if (intro && (intro->animating() || intro->isVisible()) && !main) return;
QPixmap bg = anim ? myGrab(this, QRect(0, st::titleHeight, width(), height() - st::titleHeight)) : QPixmap(); QPixmap bg = anim ? myGrab(this, QRect(0, st::titleHeight, width(), height() - st::titleHeight)) : QPixmap();

View File

@ -11,7 +11,7 @@
<key>CFBundlePackageType</key> <key>CFBundlePackageType</key>
<string>APPL</string> <string>APPL</string>
<key>CFBundleShortVersionString</key> <key>CFBundleShortVersionString</key>
<string>0.8.24</string> <string>0.8.25</string>
<key>LSMinimumSystemVersion</key> <key>LSMinimumSystemVersion</key>
<string>$(MACOSX_DEPLOYMENT_TARGET)</string> <string>$(MACOSX_DEPLOYMENT_TARGET)</string>
<key>CFBundleSignature</key> <key>CFBundleSignature</key>

Binary file not shown.

View File

@ -1703,7 +1703,7 @@
buildSettings = { buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
COPY_PHASE_STRIP = NO; COPY_PHASE_STRIP = NO;
CURRENT_PROJECT_VERSION = 0.8.24; CURRENT_PROJECT_VERSION = 0.8.25;
DEBUG_INFORMATION_FORMAT = dwarf; DEBUG_INFORMATION_FORMAT = dwarf;
GCC_GENERATE_DEBUGGING_SYMBOLS = YES; GCC_GENERATE_DEBUGGING_SYMBOLS = YES;
GCC_OPTIMIZATION_LEVEL = 0; GCC_OPTIMIZATION_LEVEL = 0;
@ -1721,7 +1721,7 @@
buildSettings = { buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
COPY_PHASE_STRIP = YES; COPY_PHASE_STRIP = YES;
CURRENT_PROJECT_VERSION = 0.8.24; CURRENT_PROJECT_VERSION = 0.8.25;
GCC_GENERATE_DEBUGGING_SYMBOLS = NO; GCC_GENERATE_DEBUGGING_SYMBOLS = NO;
GCC_OPTIMIZATION_LEVEL = fast; GCC_OPTIMIZATION_LEVEL = fast;
GCC_PREFIX_HEADER = ./SourceFiles/stdafx.h; GCC_PREFIX_HEADER = ./SourceFiles/stdafx.h;
@ -1747,10 +1747,10 @@
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
CODE_SIGN_IDENTITY = ""; CODE_SIGN_IDENTITY = "";
COPY_PHASE_STRIP = NO; COPY_PHASE_STRIP = NO;
CURRENT_PROJECT_VERSION = 0.8.24; CURRENT_PROJECT_VERSION = 0.8.25;
DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
DYLIB_COMPATIBILITY_VERSION = 0.8; DYLIB_COMPATIBILITY_VERSION = 0.8;
DYLIB_CURRENT_VERSION = 0.8.24; DYLIB_CURRENT_VERSION = 0.8.25;
ENABLE_STRICT_OBJC_MSGSEND = YES; ENABLE_STRICT_OBJC_MSGSEND = YES;
FRAMEWORK_SEARCH_PATHS = ""; FRAMEWORK_SEARCH_PATHS = "";
GCC_GENERATE_DEBUGGING_SYMBOLS = YES; GCC_GENERATE_DEBUGGING_SYMBOLS = YES;
@ -1890,10 +1890,10 @@
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
CODE_SIGN_IDENTITY = ""; CODE_SIGN_IDENTITY = "";
COPY_PHASE_STRIP = NO; COPY_PHASE_STRIP = NO;
CURRENT_PROJECT_VERSION = 0.8.24; CURRENT_PROJECT_VERSION = 0.8.25;
DEBUG_INFORMATION_FORMAT = dwarf; DEBUG_INFORMATION_FORMAT = dwarf;
DYLIB_COMPATIBILITY_VERSION = 0.8; DYLIB_COMPATIBILITY_VERSION = 0.8;
DYLIB_CURRENT_VERSION = 0.8.24; DYLIB_CURRENT_VERSION = 0.8.25;
ENABLE_STRICT_OBJC_MSGSEND = YES; ENABLE_STRICT_OBJC_MSGSEND = YES;
FRAMEWORK_SEARCH_PATHS = ""; FRAMEWORK_SEARCH_PATHS = "";
GCC_GENERATE_DEBUGGING_SYMBOLS = YES; GCC_GENERATE_DEBUGGING_SYMBOLS = YES;

View File

@ -1,2 +1,2 @@
echo 8024 0.8.24 0 echo 8025 0.8.25 1
# AppVersion AppVersionStr DevChannel # AppVersion AppVersionStr DevChannel