mirror of https://github.com/procxx/kepka.git
Use emoji/stickers/gifs slider in EmojiPan.
Also rename EmojiTabs to EmojiSections.
This commit is contained in:
parent
3d846fcd49
commit
1540f6f528
Binary file not shown.
Before Width: | Height: | Size: 518 B |
Binary file not shown.
Before Width: | Height: | Size: 1.0 KiB |
|
@ -776,6 +776,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
"lng_switch_stickers" = "Stickers";
|
"lng_switch_stickers" = "Stickers";
|
||||||
"lng_switch_stickers_gifs" = "GIFs & Stickers";
|
"lng_switch_stickers_gifs" = "GIFs & Stickers";
|
||||||
"lng_switch_emoji" = "Emoji";
|
"lng_switch_emoji" = "Emoji";
|
||||||
|
"lng_switch_gifs" = "GIFs";
|
||||||
"lng_stickers_featured_add" = "Add";
|
"lng_stickers_featured_add" = "Add";
|
||||||
|
|
||||||
"lng_saved_gifs" = "Saved GIFs";
|
"lng_saved_gifs" = "Saved GIFs";
|
||||||
|
|
|
@ -24,11 +24,60 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
#include "storage/file_download.h"
|
#include "storage/file_download.h"
|
||||||
#include "window/notifications_manager.h"
|
#include "window/notifications_manager.h"
|
||||||
|
|
||||||
|
QByteArray AuthSessionData::serialize() const {
|
||||||
|
auto size = sizeof(qint32) * 2;
|
||||||
|
|
||||||
|
auto result = QByteArray();
|
||||||
|
result.reserve(size);
|
||||||
|
{
|
||||||
|
QBuffer buffer(&result);
|
||||||
|
if (!buffer.open(QIODevice::WriteOnly)) {
|
||||||
|
Unexpected("Can't open data for AuthSessionData::serialize()");
|
||||||
|
}
|
||||||
|
|
||||||
|
QDataStream stream(&buffer);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
stream << static_cast<qint32>(_variables.emojiPanTab);
|
||||||
|
stream << qint32(_variables.lastSeenWarningSeen ? 1 : 0);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AuthSessionData::constructFromSerialized(const QByteArray &serialized) {
|
||||||
|
if (serialized.isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto readonly = serialized;
|
||||||
|
QBuffer buffer(&readonly);
|
||||||
|
if (!buffer.open(QIODevice::ReadOnly)) {
|
||||||
|
Unexpected("Can't open data for DcOptions::constructFromSerialized()");
|
||||||
|
}
|
||||||
|
QDataStream stream(&buffer);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
qint32 emojiPanTab = static_cast<qint32>(EmojiPanTabType::Emoji);
|
||||||
|
qint32 lastSeenWarningSeen = 0;
|
||||||
|
stream >> emojiPanTab;
|
||||||
|
stream >> lastSeenWarningSeen;
|
||||||
|
if (stream.status() != QDataStream::Ok) {
|
||||||
|
LOG(("App Error: Bad data for AuthSessionData::constructFromSerialized()"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto uncheckedTab = static_cast<EmojiPanTabType>(emojiPanTab);
|
||||||
|
switch (uncheckedTab) {
|
||||||
|
case EmojiPanTabType::Emoji:
|
||||||
|
case EmojiPanTabType::Stickers:
|
||||||
|
case EmojiPanTabType::Gifs: _variables.emojiPanTab = uncheckedTab; break;
|
||||||
|
}
|
||||||
|
_variables.lastSeenWarningSeen = (lastSeenWarningSeen == 1);
|
||||||
|
}
|
||||||
|
|
||||||
AuthSession::AuthSession(UserId userId)
|
AuthSession::AuthSession(UserId userId)
|
||||||
: _userId(userId)
|
: _userId(userId)
|
||||||
, _downloader(std::make_unique<Storage::Downloader>())
|
, _downloader(std::make_unique<Storage::Downloader>())
|
||||||
, _notifications(std::make_unique<Window::Notifications::System>(this)) {
|
, _notifications(std::make_unique<Window::Notifications::System>(this)) {
|
||||||
t_assert(_userId != 0);
|
Expects(_userId != 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AuthSession::Exists() {
|
bool AuthSession::Exists() {
|
||||||
|
|
|
@ -30,6 +30,56 @@ class System;
|
||||||
} // namespace Notifications
|
} // namespace Notifications
|
||||||
} // namespace Window
|
} // namespace Window
|
||||||
|
|
||||||
|
enum class EmojiPanTabType {
|
||||||
|
Emoji,
|
||||||
|
Stickers,
|
||||||
|
Gifs,
|
||||||
|
};
|
||||||
|
|
||||||
|
class AuthSessionData final {
|
||||||
|
public:
|
||||||
|
base::Variable<bool> &contactsLoaded() {
|
||||||
|
return _contactsLoaded;
|
||||||
|
}
|
||||||
|
base::Variable<bool> &allChatsLoaded() {
|
||||||
|
return _allChatsLoaded;
|
||||||
|
}
|
||||||
|
base::Observable<void> &moreChatsLoaded() {
|
||||||
|
return _moreChatsLoaded;
|
||||||
|
}
|
||||||
|
|
||||||
|
void copyFrom(const AuthSessionData &other) {
|
||||||
|
_variables = other._variables;
|
||||||
|
}
|
||||||
|
QByteArray serialize() const;
|
||||||
|
void constructFromSerialized(const QByteArray &serialized);
|
||||||
|
|
||||||
|
bool lastSeenWarningSeen() const {
|
||||||
|
return _variables.lastSeenWarningSeen;
|
||||||
|
}
|
||||||
|
void setLastSeenWarningSeen(bool lastSeenWarningSeen) {
|
||||||
|
_variables.lastSeenWarningSeen = lastSeenWarningSeen;
|
||||||
|
}
|
||||||
|
EmojiPanTabType emojiPanTab() const {
|
||||||
|
return _variables.emojiPanTab;
|
||||||
|
}
|
||||||
|
void setEmojiPanTab(EmojiPanTabType tab) {
|
||||||
|
_variables.emojiPanTab = tab;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
struct Variables {
|
||||||
|
bool lastSeenWarningSeen = false;
|
||||||
|
EmojiPanTabType emojiPanTab = EmojiPanTabType::Emoji;
|
||||||
|
};
|
||||||
|
|
||||||
|
base::Variable<bool> _contactsLoaded = { false };
|
||||||
|
base::Variable<bool> _allChatsLoaded = { false };
|
||||||
|
base::Observable<void> _moreChatsLoaded;
|
||||||
|
Variables _variables;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
class AuthSession final {
|
class AuthSession final {
|
||||||
public:
|
public:
|
||||||
AuthSession(UserId userId);
|
AuthSession(UserId userId);
|
||||||
|
@ -63,33 +113,7 @@ public:
|
||||||
return *_notifications;
|
return *_notifications;
|
||||||
}
|
}
|
||||||
|
|
||||||
class Data {
|
AuthSessionData &data() {
|
||||||
public:
|
|
||||||
base::Variable<bool> &contactsLoaded() {
|
|
||||||
return _contactsLoaded;
|
|
||||||
}
|
|
||||||
base::Variable<bool> &allChatsLoaded() {
|
|
||||||
return _allChatsLoaded;
|
|
||||||
}
|
|
||||||
base::Observable<void> &moreChatsLoaded() {
|
|
||||||
return _moreChatsLoaded;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool lastSeenWarningSeen() const {
|
|
||||||
return _lastSeenWarningSeen;
|
|
||||||
}
|
|
||||||
void setLastSeenWarningSeen(bool lastSeenWarningSeen) {
|
|
||||||
_lastSeenWarningSeen = lastSeenWarningSeen;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
base::Variable<bool> _contactsLoaded = { false } ;
|
|
||||||
base::Variable<bool> _allChatsLoaded = { false };
|
|
||||||
base::Observable<void> _moreChatsLoaded;
|
|
||||||
bool _lastSeenWarningSeen = false;
|
|
||||||
|
|
||||||
};
|
|
||||||
Data &data() {
|
|
||||||
return _data;
|
return _data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -97,7 +121,7 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
UserId _userId = 0;
|
UserId _userId = 0;
|
||||||
Data _data;
|
AuthSessionData _data;
|
||||||
|
|
||||||
const std::unique_ptr<Storage::Downloader> _downloader;
|
const std::unique_ptr<Storage::Downloader> _downloader;
|
||||||
const std::unique_ptr<Window::Notifications::System> _notifications;
|
const std::unique_ptr<Window::Notifications::System> _notifications;
|
||||||
|
|
|
@ -386,38 +386,40 @@ void StickersBox::switchTab() {
|
||||||
newTab = &_archived;
|
newTab = &_archived;
|
||||||
requestArchivedSets();
|
requestArchivedSets();
|
||||||
}
|
}
|
||||||
if (_tab != newTab) {
|
if (_tab == newTab) {
|
||||||
if (_tab == &_installed) {
|
return;
|
||||||
_localOrder = _tab->widget()->getFullOrder();
|
|
||||||
_localRemoved = _tab->widget()->getRemovedSets();
|
|
||||||
}
|
|
||||||
auto wasCache = grabContentCache();
|
|
||||||
auto wasIndex = _tab->index();
|
|
||||||
_tab->saveScrollTop();
|
|
||||||
auto widget = takeInnerWidget<Inner>();
|
|
||||||
widget->setParent(this);
|
|
||||||
widget->hide();
|
|
||||||
_tab->returnWidget(std::move(widget));
|
|
||||||
_tab = newTab;
|
|
||||||
_section = newSection;
|
|
||||||
setInnerWidget(_tab->takeWidget(), getTopSkip());
|
|
||||||
_tabs->raise();
|
|
||||||
_unreadBadge->raise();
|
|
||||||
_tab->widget()->show();
|
|
||||||
rebuildList();
|
|
||||||
onScrollToY(_tab->getScrollTop());
|
|
||||||
auto nowCache = grabContentCache();
|
|
||||||
auto nowIndex = _tab->index();
|
|
||||||
|
|
||||||
_slideAnimation = std::make_unique<Ui::SlideAnimation>();
|
|
||||||
_slideAnimation->setSnapshots(std::move(wasCache), std::move(nowCache));
|
|
||||||
auto slideLeft = wasIndex > nowIndex;
|
|
||||||
_slideAnimation->start(slideLeft, [this] { update(); }, st::slideDuration);
|
|
||||||
setInnerVisible(false);
|
|
||||||
|
|
||||||
setFocus();
|
|
||||||
update();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (_tab == &_installed) {
|
||||||
|
_localOrder = _tab->widget()->getFullOrder();
|
||||||
|
_localRemoved = _tab->widget()->getRemovedSets();
|
||||||
|
}
|
||||||
|
auto wasCache = grabContentCache();
|
||||||
|
auto wasIndex = _tab->index();
|
||||||
|
_tab->saveScrollTop();
|
||||||
|
auto widget = takeInnerWidget<Inner>();
|
||||||
|
widget->setParent(this);
|
||||||
|
widget->hide();
|
||||||
|
_tab->returnWidget(std::move(widget));
|
||||||
|
_tab = newTab;
|
||||||
|
_section = newSection;
|
||||||
|
setInnerWidget(_tab->takeWidget(), getTopSkip());
|
||||||
|
_tabs->raise();
|
||||||
|
_unreadBadge->raise();
|
||||||
|
_tab->widget()->show();
|
||||||
|
rebuildList();
|
||||||
|
onScrollToY(_tab->getScrollTop());
|
||||||
|
auto nowCache = grabContentCache();
|
||||||
|
auto nowIndex = _tab->index();
|
||||||
|
|
||||||
|
_slideAnimation = std::make_unique<Ui::SlideAnimation>();
|
||||||
|
_slideAnimation->setSnapshots(std::move(wasCache), std::move(nowCache));
|
||||||
|
auto slideLeft = wasIndex > nowIndex;
|
||||||
|
_slideAnimation->start(slideLeft, [this] { update(); }, st::slideDuration);
|
||||||
|
setInnerVisible(false);
|
||||||
|
|
||||||
|
setFocus();
|
||||||
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
QPixmap StickersBox::grabContentCache() {
|
QPixmap StickersBox::grabContentCache() {
|
||||||
|
|
|
@ -35,6 +35,12 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
#include "auth_session.h"
|
#include "auth_session.h"
|
||||||
#include "messenger.h"
|
#include "messenger.h"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
constexpr auto kStickerPanPerRow = Stickers::kPanPerRow;
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
StickerSetBox::StickerSetBox(QWidget*, const MTPInputStickerSet &set)
|
StickerSetBox::StickerSetBox(QWidget*, const MTPInputStickerSet &set)
|
||||||
: _set(set) {
|
: _set(set) {
|
||||||
}
|
}
|
||||||
|
@ -181,8 +187,8 @@ void StickerSetBox::Inner::gotSet(const MTPmessages_StickerSet &set) {
|
||||||
if (_pack.isEmpty()) {
|
if (_pack.isEmpty()) {
|
||||||
Ui::show(Box<InformBox>(lang(lng_stickers_not_found)));
|
Ui::show(Box<InformBox>(lang(lng_stickers_not_found)));
|
||||||
} else {
|
} else {
|
||||||
int32 rows = _pack.size() / StickerPanPerRow + ((_pack.size() % StickerPanPerRow) ? 1 : 0);
|
int32 rows = _pack.size() / kStickerPanPerRow + ((_pack.size() % kStickerPanPerRow) ? 1 : 0);
|
||||||
resize(st::stickersPadding.left() + StickerPanPerRow * st::stickersSize.width(), st::stickersPadding.top() + rows * st::stickersSize.height() + st::stickersPadding.bottom());
|
resize(st::stickersPadding.left() + kStickerPanPerRow * st::stickersSize.width(), st::stickersPadding.top() + rows * st::stickersSize.height() + st::stickersPadding.bottom());
|
||||||
}
|
}
|
||||||
_loaded = true;
|
_loaded = true;
|
||||||
|
|
||||||
|
@ -319,8 +325,8 @@ void StickerSetBox::Inner::setSelected(int selected) {
|
||||||
void StickerSetBox::Inner::startOverAnimation(int index, float64 from, float64 to) {
|
void StickerSetBox::Inner::startOverAnimation(int index, float64 from, float64 to) {
|
||||||
if (index >= 0 && index < _packOvers.size()) {
|
if (index >= 0 && index < _packOvers.size()) {
|
||||||
_packOvers[index].start([this, index] {
|
_packOvers[index].start([this, index] {
|
||||||
int row = index / StickerPanPerRow;
|
int row = index / kStickerPanPerRow;
|
||||||
int column = index % StickerPanPerRow;
|
int column = index % kStickerPanPerRow;
|
||||||
int left = st::stickersPadding.left() + column * st::stickersSize.width();
|
int left = st::stickersPadding.left() + column * st::stickersSize.width();
|
||||||
int top = st::stickersPadding.top() + row * st::stickersSize.height();
|
int top = st::stickersPadding.top() + row * st::stickersSize.height();
|
||||||
rtlupdate(left, top, st::stickersSize.width(), st::stickersSize.height());
|
rtlupdate(left, top, st::stickersSize.width(), st::stickersSize.height());
|
||||||
|
@ -341,8 +347,8 @@ int32 StickerSetBox::Inner::stickerFromGlobalPos(const QPoint &p) const {
|
||||||
if (rtl()) l.setX(width() - l.x());
|
if (rtl()) l.setX(width() - l.x());
|
||||||
int32 row = (l.y() >= st::stickersPadding.top()) ? qFloor((l.y() - st::stickersPadding.top()) / st::stickersSize.height()) : -1;
|
int32 row = (l.y() >= st::stickersPadding.top()) ? qFloor((l.y() - st::stickersPadding.top()) / st::stickersSize.height()) : -1;
|
||||||
int32 col = (l.x() >= st::stickersPadding.left()) ? qFloor((l.x() - st::stickersPadding.left()) / st::stickersSize.width()) : -1;
|
int32 col = (l.x() >= st::stickersPadding.left()) ? qFloor((l.x() - st::stickersPadding.left()) / st::stickersSize.width()) : -1;
|
||||||
if (row >= 0 && col >= 0 && col < StickerPanPerRow) {
|
if (row >= 0 && col >= 0 && col < kStickerPanPerRow) {
|
||||||
int32 result = row * StickerPanPerRow + col;
|
int32 result = row * kStickerPanPerRow + col;
|
||||||
return (result < _pack.size()) ? result : -1;
|
return (result < _pack.size()) ? result : -1;
|
||||||
}
|
}
|
||||||
return -1;
|
return -1;
|
||||||
|
@ -355,12 +361,12 @@ void StickerSetBox::Inner::paintEvent(QPaintEvent *e) {
|
||||||
if (_pack.isEmpty()) return;
|
if (_pack.isEmpty()) return;
|
||||||
|
|
||||||
auto ms = getms();
|
auto ms = getms();
|
||||||
int32 rows = _pack.size() / StickerPanPerRow + ((_pack.size() % StickerPanPerRow) ? 1 : 0);
|
int32 rows = _pack.size() / kStickerPanPerRow + ((_pack.size() % kStickerPanPerRow) ? 1 : 0);
|
||||||
int32 from = qFloor(e->rect().top() / st::stickersSize.height()), to = qFloor(e->rect().bottom() / st::stickersSize.height()) + 1;
|
int32 from = qFloor(e->rect().top() / st::stickersSize.height()), to = qFloor(e->rect().bottom() / st::stickersSize.height()) + 1;
|
||||||
|
|
||||||
for (int32 i = from; i < to; ++i) {
|
for (int32 i = from; i < to; ++i) {
|
||||||
for (int32 j = 0; j < StickerPanPerRow; ++j) {
|
for (int32 j = 0; j < kStickerPanPerRow; ++j) {
|
||||||
int32 index = i * StickerPanPerRow + j;
|
int32 index = i * kStickerPanPerRow + j;
|
||||||
if (index >= _pack.size()) break;
|
if (index >= _pack.size()) break;
|
||||||
t_assert(index < _packOvers.size());
|
t_assert(index < _packOvers.size());
|
||||||
|
|
||||||
|
|
|
@ -393,18 +393,18 @@ void Init() {\n\
|
||||||
|
|
||||||
bool Generator::writePacks() {
|
bool Generator::writePacks() {
|
||||||
constexpr const char *packNames[] = {
|
constexpr const char *packNames[] = {
|
||||||
"dbietPeople",
|
"dbiesPeople",
|
||||||
"dbietNature",
|
"dbiesNature",
|
||||||
"dbietFood",
|
"dbiesFood",
|
||||||
"dbietActivity",
|
"dbiesActivity",
|
||||||
"dbietTravel",
|
"dbiesTravel",
|
||||||
"dbietObjects",
|
"dbiesObjects",
|
||||||
"dbietSymbols",
|
"dbiesSymbols",
|
||||||
};
|
};
|
||||||
source_->stream() << "\
|
source_->stream() << "\
|
||||||
\n\
|
\n\
|
||||||
int GetPackCount(DBIEmojiTab tab) {\n\
|
int GetPackCount(DBIEmojiSection section) {\n\
|
||||||
switch (tab) {\n";
|
switch (section) {\n";
|
||||||
auto countIndex = 0;
|
auto countIndex = 0;
|
||||||
for (auto name : packNames) {
|
for (auto name : packNames) {
|
||||||
if (countIndex >= int(data_.categories.size())) {
|
if (countIndex >= int(data_.categories.size())) {
|
||||||
|
@ -415,13 +415,13 @@ int GetPackCount(DBIEmojiTab tab) {\n\
|
||||||
case " << name << ": return " << data_.categories[countIndex++].size() << ";\n";
|
case " << name << ": return " << data_.categories[countIndex++].size() << ";\n";
|
||||||
}
|
}
|
||||||
source_->stream() << "\
|
source_->stream() << "\
|
||||||
case dbietRecent: return cGetRecentEmoji().size();\n\
|
case dbiesRecent: return GetRecent().size();\n\
|
||||||
}\n\
|
}\n\
|
||||||
return 0;\n\
|
return 0;\n\
|
||||||
}\n\
|
}\n\
|
||||||
\n\
|
\n\
|
||||||
EmojiPack GetPack(DBIEmojiTab tab) {\n\
|
EmojiPack GetPack(DBIEmojiSection section) {\n\
|
||||||
switch (tab) {\n";
|
switch (section) {\n";
|
||||||
auto index = 0;
|
auto index = 0;
|
||||||
for (auto name : packNames) {
|
for (auto name : packNames) {
|
||||||
if (index >= int(data_.categories.size())) {
|
if (index >= int(data_.categories.size())) {
|
||||||
|
@ -444,10 +444,10 @@ EmojiPack GetPack(DBIEmojiTab tab) {\n\
|
||||||
} break;\n\n";
|
} break;\n\n";
|
||||||
}
|
}
|
||||||
source_->stream() << "\
|
source_->stream() << "\
|
||||||
case dbietRecent: {\n\
|
case dbiesRecent: {\n\
|
||||||
auto result = EmojiPack();\n\
|
auto result = EmojiPack();\n\
|
||||||
result.reserve(cGetRecentEmoji().size());\n\
|
result.reserve(GetRecent().size());\n\
|
||||||
for (auto &item : cGetRecentEmoji()) {\n\
|
for (auto &item : GetRecent()) {\n\
|
||||||
result.push_back(item.first);\n\
|
result.push_back(item.first);\n\
|
||||||
}\n\
|
}\n\
|
||||||
return result;\n\
|
return result;\n\
|
||||||
|
|
|
@ -110,11 +110,6 @@ enum {
|
||||||
ShortcutsCountLimit = 256, // how many shortcuts can be in json file
|
ShortcutsCountLimit = 256, // how many shortcuts can be in json file
|
||||||
|
|
||||||
PreloadHeightsCount = 3, // when 3 screens to scroll left make a preload request
|
PreloadHeightsCount = 3, // when 3 screens to scroll left make a preload request
|
||||||
EmojiPanPerRow = 7,
|
|
||||||
EmojiPanRowsPerPage = 6,
|
|
||||||
StickerPanPerRow = 5,
|
|
||||||
StickerPanRowsPerPage = 4,
|
|
||||||
StickersUpdateTimeout = 3600000, // update not more than once in an hour
|
|
||||||
|
|
||||||
SearchPeopleLimit = 5,
|
SearchPeopleLimit = 5,
|
||||||
MinUsernameLength = 5,
|
MinUsernameLength = 5,
|
||||||
|
|
|
@ -223,6 +223,12 @@ inline QFlags<Enum> qFlags(Enum v) {
|
||||||
return QFlags<Enum>(v);
|
return QFlags<Enum>(v);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename Lambda>
|
||||||
|
inline void InvokeQueued(QObject *context, Lambda &&lambda) {
|
||||||
|
QObject proxy;
|
||||||
|
QObject::connect(&proxy, &QObject::destroyed, context, std::forward<Lambda>(lambda), Qt::QueuedConnection);
|
||||||
|
}
|
||||||
|
|
||||||
static const int32 ScrollMax = INT_MAX;
|
static const int32 ScrollMax = INT_MAX;
|
||||||
|
|
||||||
extern uint64 _SharedMemoryLocation[];
|
extern uint64 _SharedMemoryLocation[];
|
||||||
|
@ -521,21 +527,17 @@ enum DBIScale {
|
||||||
|
|
||||||
static const int MatrixRowShift = 40000;
|
static const int MatrixRowShift = 40000;
|
||||||
|
|
||||||
enum DBIEmojiTab {
|
enum DBIEmojiSection {
|
||||||
dbietRecent = -1,
|
dbiesRecent = -1,
|
||||||
dbietPeople = 0,
|
dbiesPeople = 0,
|
||||||
dbietNature = 1,
|
dbiesNature = 1,
|
||||||
dbietFood = 2,
|
dbiesFood = 2,
|
||||||
dbietActivity = 3,
|
dbiesActivity = 3,
|
||||||
dbietTravel = 4,
|
dbiesTravel = 4,
|
||||||
dbietObjects = 5,
|
dbiesObjects = 5,
|
||||||
dbietSymbols = 6,
|
dbiesSymbols = 6,
|
||||||
dbietStickers = 666,
|
dbiesStickers = 666,
|
||||||
};
|
};
|
||||||
static const int emojiTabCount = 8;
|
|
||||||
inline DBIEmojiTab emojiTabAtIndex(int index) {
|
|
||||||
return (index < 0 || index >= emojiTabCount) ? dbietRecent : DBIEmojiTab(index - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
enum DBIPlatform {
|
enum DBIPlatform {
|
||||||
dbipWindows = 0,
|
dbipWindows = 0,
|
||||||
|
|
|
@ -67,6 +67,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
|
constexpr auto kStickersUpdateTimeout = 3600000; // update not more than once in an hour
|
||||||
|
|
||||||
QString mimeTagFromTag(const QString &tagId) {
|
QString mimeTagFromTag(const QString &tagId) {
|
||||||
if (tagId.startsWith(qstr("mention://"))) {
|
if (tagId.startsWith(qstr("mention://"))) {
|
||||||
return tagId + ':' + QString::number(AuthSession::CurrentUserId());
|
return tagId + ':' + QString::number(AuthSession::CurrentUserId());
|
||||||
|
@ -3718,22 +3720,22 @@ void HistoryWidget::onRecordUpdate(quint16 level, qint32 samples) {
|
||||||
|
|
||||||
void HistoryWidget::updateStickers() {
|
void HistoryWidget::updateStickers() {
|
||||||
auto now = getms(true);
|
auto now = getms(true);
|
||||||
if (!Global::LastStickersUpdate() || now >= Global::LastStickersUpdate() + StickersUpdateTimeout) {
|
if (!Global::LastStickersUpdate() || now >= Global::LastStickersUpdate() + kStickersUpdateTimeout) {
|
||||||
if (!_stickersUpdateRequest) {
|
if (!_stickersUpdateRequest) {
|
||||||
_stickersUpdateRequest = MTP::send(MTPmessages_GetAllStickers(MTP_int(Local::countStickersHash(true))), rpcDone(&HistoryWidget::stickersGot), rpcFail(&HistoryWidget::stickersFailed));
|
_stickersUpdateRequest = MTP::send(MTPmessages_GetAllStickers(MTP_int(Local::countStickersHash(true))), rpcDone(&HistoryWidget::stickersGot), rpcFail(&HistoryWidget::stickersFailed));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!Global::LastRecentStickersUpdate() || now >= Global::LastRecentStickersUpdate() + StickersUpdateTimeout) {
|
if (!Global::LastRecentStickersUpdate() || now >= Global::LastRecentStickersUpdate() + kStickersUpdateTimeout) {
|
||||||
if (!_recentStickersUpdateRequest) {
|
if (!_recentStickersUpdateRequest) {
|
||||||
_recentStickersUpdateRequest = MTP::send(MTPmessages_GetRecentStickers(MTP_flags(0), MTP_int(Local::countRecentStickersHash())), rpcDone(&HistoryWidget::recentStickersGot), rpcFail(&HistoryWidget::recentStickersFailed));
|
_recentStickersUpdateRequest = MTP::send(MTPmessages_GetRecentStickers(MTP_flags(0), MTP_int(Local::countRecentStickersHash())), rpcDone(&HistoryWidget::recentStickersGot), rpcFail(&HistoryWidget::recentStickersFailed));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!Global::LastFeaturedStickersUpdate() || now >= Global::LastFeaturedStickersUpdate() + StickersUpdateTimeout) {
|
if (!Global::LastFeaturedStickersUpdate() || now >= Global::LastFeaturedStickersUpdate() + kStickersUpdateTimeout) {
|
||||||
if (!_featuredStickersUpdateRequest) {
|
if (!_featuredStickersUpdateRequest) {
|
||||||
_featuredStickersUpdateRequest = MTP::send(MTPmessages_GetFeaturedStickers(MTP_int(Local::countFeaturedStickersHash())), rpcDone(&HistoryWidget::featuredStickersGot), rpcFail(&HistoryWidget::featuredStickersFailed));
|
_featuredStickersUpdateRequest = MTP::send(MTPmessages_GetFeaturedStickers(MTP_int(Local::countFeaturedStickersHash())), rpcDone(&HistoryWidget::featuredStickersGot), rpcFail(&HistoryWidget::featuredStickersFailed));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!cLastSavedGifsUpdate() || now >= cLastSavedGifsUpdate() + StickersUpdateTimeout) {
|
if (!cLastSavedGifsUpdate() || now >= cLastSavedGifsUpdate() + kStickersUpdateTimeout) {
|
||||||
if (!_savedGifsUpdateRequest) {
|
if (!_savedGifsUpdateRequest) {
|
||||||
_savedGifsUpdateRequest = MTP::send(MTPmessages_GetSavedGifs(MTP_int(Local::countSavedGifsHash())), rpcDone(&HistoryWidget::savedGifsGot), rpcFail(&HistoryWidget::savedGifsFailed));
|
_savedGifsUpdateRequest = MTP::send(MTPmessages_GetSavedGifs(MTP_int(Local::countSavedGifsHash())), rpcDone(&HistoryWidget::savedGifsGot), rpcFail(&HistoryWidget::savedGifsFailed));
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,6 +21,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
#include "settings.h"
|
#include "settings.h"
|
||||||
|
|
||||||
#include "platform/platform_specific.h"
|
#include "platform/platform_specific.h"
|
||||||
|
#include "stickers/emoji_pan.h"
|
||||||
#include "lang.h"
|
#include "lang.h"
|
||||||
|
|
||||||
bool gRtl = false;
|
bool gRtl = false;
|
||||||
|
@ -82,7 +83,6 @@ RecentStickerPack gRecentStickers;
|
||||||
|
|
||||||
SavedGifs gSavedGifs;
|
SavedGifs gSavedGifs;
|
||||||
TimeMs gLastSavedGifsUpdate = 0;
|
TimeMs gLastSavedGifsUpdate = 0;
|
||||||
bool gShowingSavedGifs = false;
|
|
||||||
|
|
||||||
RecentHashtagPack gRecentWriteHashtags, gRecentSearchHashtags;
|
RecentHashtagPack gRecentWriteHashtags, gRecentSearchHashtags;
|
||||||
|
|
||||||
|
@ -212,79 +212,6 @@ void settingsParseArgs(int argc, char *argv[]) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
RecentEmojiPack &cGetRecentEmoji() {
|
|
||||||
if (cRecentEmoji().isEmpty()) {
|
|
||||||
RecentEmojiPack result;
|
|
||||||
auto haveAlready = [&result](EmojiPtr emoji) {
|
|
||||||
for (auto &row : result) {
|
|
||||||
if (row.first->id() == emoji->id()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
if (!cRecentEmojiPreload().isEmpty()) {
|
|
||||||
auto preload = cRecentEmojiPreload();
|
|
||||||
cSetRecentEmojiPreload(RecentEmojiPreload());
|
|
||||||
result.reserve(preload.size());
|
|
||||||
for (auto i = preload.cbegin(), e = preload.cend(); i != e; ++i) {
|
|
||||||
if (auto emoji = Ui::Emoji::Find(i->first)) {
|
|
||||||
if (!haveAlready(emoji)) {
|
|
||||||
result.push_back(qMakePair(emoji, i->second));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
uint64 defaultRecent[] = {
|
|
||||||
0xD83DDE02LLU,
|
|
||||||
0xD83DDE18LLU,
|
|
||||||
0x2764LLU,
|
|
||||||
0xD83DDE0DLLU,
|
|
||||||
0xD83DDE0ALLU,
|
|
||||||
0xD83DDE01LLU,
|
|
||||||
0xD83DDC4DLLU,
|
|
||||||
0x263ALLU,
|
|
||||||
0xD83DDE14LLU,
|
|
||||||
0xD83DDE04LLU,
|
|
||||||
0xD83DDE2DLLU,
|
|
||||||
0xD83DDC8BLLU,
|
|
||||||
0xD83DDE12LLU,
|
|
||||||
0xD83DDE33LLU,
|
|
||||||
0xD83DDE1CLLU,
|
|
||||||
0xD83DDE48LLU,
|
|
||||||
0xD83DDE09LLU,
|
|
||||||
0xD83DDE03LLU,
|
|
||||||
0xD83DDE22LLU,
|
|
||||||
0xD83DDE1DLLU,
|
|
||||||
0xD83DDE31LLU,
|
|
||||||
0xD83DDE21LLU,
|
|
||||||
0xD83DDE0FLLU,
|
|
||||||
0xD83DDE1ELLU,
|
|
||||||
0xD83DDE05LLU,
|
|
||||||
0xD83DDE1ALLU,
|
|
||||||
0xD83DDE4ALLU,
|
|
||||||
0xD83DDE0CLLU,
|
|
||||||
0xD83DDE00LLU,
|
|
||||||
0xD83DDE0BLLU,
|
|
||||||
0xD83DDE06LLU,
|
|
||||||
0xD83DDC4CLLU,
|
|
||||||
0xD83DDE10LLU,
|
|
||||||
0xD83DDE15LLU,
|
|
||||||
};
|
|
||||||
for (auto oldKey : defaultRecent) {
|
|
||||||
if (result.size() >= EmojiPanPerRow * EmojiPanRowsPerPage) break;
|
|
||||||
|
|
||||||
if (auto emoji = Ui::Emoji::FromOldKey(oldKey)) {
|
|
||||||
if (!haveAlready(emoji)) {
|
|
||||||
result.push_back(qMakePair(emoji, 1));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
cSetRecentEmoji(result);
|
|
||||||
}
|
|
||||||
return cRefRecentEmoji();
|
|
||||||
}
|
|
||||||
|
|
||||||
RecentStickerPack &cGetRecentStickers() {
|
RecentStickerPack &cGetRecentStickers() {
|
||||||
if (cRecentStickers().isEmpty() && !cRecentStickersPreload().isEmpty()) {
|
if (cRecentStickers().isEmpty() && !cRecentStickersPreload().isEmpty()) {
|
||||||
RecentStickerPreload p(cRecentStickersPreload());
|
RecentStickerPreload p(cRecentStickersPreload());
|
||||||
|
|
|
@ -156,8 +156,6 @@ DeclareRefSetting(RecentEmojiPack, RecentEmoji);
|
||||||
DeclareSetting(RecentEmojiPreload, RecentEmojiPreload);
|
DeclareSetting(RecentEmojiPreload, RecentEmojiPreload);
|
||||||
DeclareRefSetting(EmojiColorVariants, EmojiVariants);
|
DeclareRefSetting(EmojiColorVariants, EmojiVariants);
|
||||||
|
|
||||||
RecentEmojiPack &cGetRecentEmoji();
|
|
||||||
|
|
||||||
class DocumentData;
|
class DocumentData;
|
||||||
typedef QVector<DocumentData*> StickerPack;
|
typedef QVector<DocumentData*> StickerPack;
|
||||||
|
|
||||||
|
@ -174,7 +172,6 @@ typedef QMap<EmojiPtr, StickerPack> StickersByEmojiMap;
|
||||||
typedef QVector<DocumentData*> SavedGifs;
|
typedef QVector<DocumentData*> SavedGifs;
|
||||||
DeclareRefSetting(SavedGifs, SavedGifs);
|
DeclareRefSetting(SavedGifs, SavedGifs);
|
||||||
DeclareSetting(TimeMs, LastSavedGifsUpdate);
|
DeclareSetting(TimeMs, LastSavedGifsUpdate);
|
||||||
DeclareSetting(bool, ShowingSavedGifs);
|
|
||||||
|
|
||||||
typedef QList<QPair<QString, ushort> > RecentHashtagPack;
|
typedef QList<QPair<QString, ushort> > RecentHashtagPack;
|
||||||
DeclareRefSetting(RecentHashtagPack, RecentWriteHashtags);
|
DeclareRefSetting(RecentHashtagPack, RecentWriteHashtags);
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -25,6 +25,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
#include "ui/effects/panel_animation.h"
|
#include "ui/effects/panel_animation.h"
|
||||||
#include "mtproto/sender.h"
|
#include "mtproto/sender.h"
|
||||||
#include "inline_bots/inline_bot_layout_item.h"
|
#include "inline_bots/inline_bot_layout_item.h"
|
||||||
|
#include "auth_session.h"
|
||||||
|
|
||||||
namespace InlineBots {
|
namespace InlineBots {
|
||||||
namespace Layout {
|
namespace Layout {
|
||||||
|
@ -34,16 +35,23 @@ class Result;
|
||||||
} // namespace InlineBots
|
} // namespace InlineBots
|
||||||
|
|
||||||
namespace Ui {
|
namespace Ui {
|
||||||
|
class PlainShadow;
|
||||||
class ScrollArea;
|
class ScrollArea;
|
||||||
class IconButton;
|
class IconButton;
|
||||||
class LinkButton;
|
class LinkButton;
|
||||||
class RoundButton;
|
class RoundButton;
|
||||||
class RippleAnimation;
|
class RippleAnimation;
|
||||||
|
class SettingsSlider;
|
||||||
} // namesapce Ui
|
} // namesapce Ui
|
||||||
|
|
||||||
namespace internal {
|
namespace internal {
|
||||||
|
|
||||||
constexpr int kInlineItemsMaxPerRow = 5;
|
constexpr auto kInlineItemsMaxPerRow = 5;
|
||||||
|
|
||||||
|
constexpr auto kEmojiSectionCount = 8;
|
||||||
|
inline DBIEmojiSection EmojiSectionAtIndex(int index) {
|
||||||
|
return (index < 0 || index >= kEmojiSectionCount) ? dbiesRecent : DBIEmojiSection(index - 1);
|
||||||
|
}
|
||||||
|
|
||||||
using InlineResult = InlineBots::Result;
|
using InlineResult = InlineBots::Result;
|
||||||
using InlineResults = std::vector<std::unique_ptr<InlineResult>>;
|
using InlineResults = std::vector<std::unique_ptr<InlineResult>>;
|
||||||
|
@ -109,38 +117,56 @@ private:
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class EmojiPanel;
|
class BasicPanInner : public TWidget {
|
||||||
class EmojiPanInner : public TWidget {
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
BasicPanInner(QWidget *parent);
|
||||||
|
|
||||||
|
void setVisibleTopBottom(int visibleTop, int visibleBottom) override;
|
||||||
|
|
||||||
|
int getVisibleTop() const {
|
||||||
|
return _visibleTop;
|
||||||
|
}
|
||||||
|
int getVisibleBottom() const {
|
||||||
|
return _visibleBottom;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void refreshRecent() = 0;
|
||||||
|
virtual void preloadImages() {
|
||||||
|
}
|
||||||
|
virtual void hideFinish(bool completely) = 0;
|
||||||
|
virtual void clearSelection() = 0;
|
||||||
|
|
||||||
|
virtual object_ptr<TWidget> createController() = 0;
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void scrollToY(int y);
|
||||||
|
void disableScroll(bool disabled);
|
||||||
|
void saveConfigDelayed(int delay);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual int countHeight() = 0;
|
||||||
|
|
||||||
|
private:
|
||||||
|
int _visibleTop = 0;
|
||||||
|
int _visibleBottom = 0;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class EmojiPanInner : public BasicPanInner {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
EmojiPanInner(QWidget *parent);
|
EmojiPanInner(QWidget *parent);
|
||||||
|
|
||||||
void setMaxHeight(int maxHeight);
|
void refreshRecent() override;
|
||||||
|
void hideFinish(bool completely) override;
|
||||||
|
void clearSelection() override;
|
||||||
|
object_ptr<TWidget> createController() override;
|
||||||
|
|
||||||
void hideFinish();
|
void showEmojiSection(DBIEmojiSection section);
|
||||||
|
DBIEmojiSection currentSection(int yOffset) const;
|
||||||
void showEmojiPack(DBIEmojiTab packIndex);
|
|
||||||
|
|
||||||
void clearSelection();
|
|
||||||
|
|
||||||
DBIEmojiTab currentTab(int yOffset) const;
|
|
||||||
|
|
||||||
void refreshRecent();
|
|
||||||
|
|
||||||
void setVisibleTopBottom(int visibleTop, int visibleBottom) override;
|
|
||||||
|
|
||||||
void fillPanels(QVector<EmojiPanel*> &panels);
|
|
||||||
void refreshPanels(QVector<EmojiPanel*> &panels);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void mousePressEvent(QMouseEvent *e) override;
|
|
||||||
void mouseReleaseEvent(QMouseEvent *e) override;
|
|
||||||
void mouseMoveEvent(QMouseEvent *e) override;
|
|
||||||
void paintEvent(QPaintEvent *e) override;
|
|
||||||
void leaveEventHook(QEvent *e) override;
|
|
||||||
void leaveToChildEvent(QEvent *e, QWidget *child) override;
|
|
||||||
void enterFromChildEvent(QEvent *e, QWidget *child) override;
|
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void onShowPicker();
|
void onShowPicker();
|
||||||
|
@ -151,31 +177,46 @@ public slots:
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void selected(EmojiPtr emoji);
|
void selected(EmojiPtr emoji);
|
||||||
|
|
||||||
void switchToStickers();
|
void switchToStickers();
|
||||||
|
|
||||||
void scrollToY(int y);
|
protected:
|
||||||
void disableScroll(bool dis);
|
void mousePressEvent(QMouseEvent *e) override;
|
||||||
|
void mouseReleaseEvent(QMouseEvent *e) override;
|
||||||
void needRefreshPanels();
|
void mouseMoveEvent(QMouseEvent *e) override;
|
||||||
void saveConfigDelayed(int32 delay);
|
void paintEvent(QPaintEvent *e) override;
|
||||||
|
void leaveEventHook(QEvent *e) override;
|
||||||
|
void leaveToChildEvent(QEvent *e, QWidget *child) override;
|
||||||
|
void enterFromChildEvent(QEvent *e, QWidget *child) override;
|
||||||
|
bool event(QEvent *e) override;
|
||||||
|
int countHeight() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
class Controller;
|
||||||
|
|
||||||
|
struct SectionInfo {
|
||||||
|
int section = 0;
|
||||||
|
int count = 0;
|
||||||
|
int top = 0;
|
||||||
|
int rowsCount = 0;
|
||||||
|
int rowsTop = 0;
|
||||||
|
int rowsBottom = 0;
|
||||||
|
};
|
||||||
|
template <typename Callback>
|
||||||
|
bool enumerateSections(Callback callback) const;
|
||||||
|
SectionInfo sectionInfo(int section) const;
|
||||||
|
SectionInfo sectionInfoByOffset(int yOffset) const;
|
||||||
|
|
||||||
|
void ensureLoaded(int section);
|
||||||
|
int countSectionTop(int section) const;
|
||||||
void updateSelected();
|
void updateSelected();
|
||||||
void setSelected(int newSelected);
|
void setSelected(int newSelected);
|
||||||
|
|
||||||
int32 _maxHeight;
|
|
||||||
|
|
||||||
int countHeight();
|
|
||||||
void selectEmoji(EmojiPtr emoji);
|
void selectEmoji(EmojiPtr emoji);
|
||||||
|
|
||||||
QRect emojiRect(int tab, int sel);
|
QRect emojiRect(int section, int sel);
|
||||||
|
|
||||||
int _visibleTop = 0;
|
int _counts[kEmojiSectionCount];
|
||||||
int _visibleBottom = 0;
|
QVector<EmojiPtr> _emoji[kEmojiSectionCount];
|
||||||
int _counts[emojiTabCount];
|
|
||||||
|
|
||||||
QVector<EmojiPtr> _emojis[emojiTabCount];
|
|
||||||
|
|
||||||
int32 _esize;
|
int32 _esize;
|
||||||
|
|
||||||
|
@ -194,54 +235,44 @@ struct StickerIcon {
|
||||||
}
|
}
|
||||||
StickerIcon(uint64 setId, DocumentData *sticker, int32 pixw, int32 pixh) : setId(setId), sticker(sticker), pixw(pixw), pixh(pixh) {
|
StickerIcon(uint64 setId, DocumentData *sticker, int32 pixw, int32 pixh) : setId(setId), sticker(sticker), pixw(pixw), pixh(pixh) {
|
||||||
}
|
}
|
||||||
uint64 setId;
|
uint64 setId = 0;
|
||||||
DocumentData *sticker = nullptr;
|
DocumentData *sticker = nullptr;
|
||||||
int pixw = 0;
|
int pixw = 0;
|
||||||
int pixh = 0;
|
int pixh = 0;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class StickerPanInner : public TWidget, public InlineBots::Layout::Context, private base::Subscriber {
|
class StickerPanInner : public BasicPanInner, public InlineBots::Layout::Context, private base::Subscriber {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
StickerPanInner(QWidget *parent);
|
StickerPanInner(QWidget *parent, bool gifs);
|
||||||
|
|
||||||
void setMaxHeight(int maxHeight);
|
void refreshRecent() override;
|
||||||
|
void preloadImages() override;
|
||||||
|
void hideFinish(bool completely) override;
|
||||||
|
void clearSelection() override;
|
||||||
|
object_ptr<TWidget> createController() override;
|
||||||
|
|
||||||
void hideFinish(bool completely);
|
|
||||||
void showStickerSet(uint64 setId);
|
void showStickerSet(uint64 setId);
|
||||||
void updateShowingSavedGifs();
|
|
||||||
|
|
||||||
bool showSectionIcons() const;
|
|
||||||
void clearSelection();
|
|
||||||
|
|
||||||
void refreshStickers();
|
void refreshStickers();
|
||||||
void refreshRecentStickers(bool resize = true);
|
void refreshRecentStickers(bool resize = true);
|
||||||
void refreshSavedGifs();
|
void refreshSavedGifs();
|
||||||
int refreshInlineRows(UserData *bot, const InlineCacheEntry *results, bool resultsDeleted);
|
int refreshInlineRows(UserData *bot, const InlineCacheEntry *results, bool resultsDeleted);
|
||||||
void refreshRecent();
|
|
||||||
void inlineBotChanged();
|
void inlineBotChanged();
|
||||||
void hideInlineRowsPanel();
|
void hideInlineRowsPanel();
|
||||||
void clearInlineRowsPanel();
|
void clearInlineRowsPanel();
|
||||||
|
|
||||||
void fillIcons(QList<StickerIcon> &icons);
|
void fillIcons(QList<StickerIcon> &icons);
|
||||||
void fillPanels(QVector<EmojiPanel*> &panels);
|
|
||||||
void refreshPanels(QVector<EmojiPanel*> &panels);
|
|
||||||
|
|
||||||
void setVisibleTopBottom(int visibleTop, int visibleBottom) override;
|
void setVisibleTopBottom(int visibleTop, int visibleBottom) override;
|
||||||
void preloadImages();
|
|
||||||
|
|
||||||
uint64 currentSet(int yOffset) const;
|
uint64 currentSet(int yOffset) const;
|
||||||
|
|
||||||
void inlineItemLayoutChanged(const InlineItem *layout) override;
|
void inlineItemLayoutChanged(const InlineItem *layout) override;
|
||||||
void inlineItemRepaint(const InlineItem *layout) override;
|
void inlineItemRepaint(const InlineItem *layout) override;
|
||||||
bool inlineItemVisible(const InlineItem *layout) override;
|
bool inlineItemVisible(const InlineItem *layout) override;
|
||||||
bool ui_isInlineItemBeingChosen();
|
|
||||||
|
|
||||||
bool inlineResultsShown() const {
|
|
||||||
return (_section == Section::Inlines);
|
|
||||||
}
|
|
||||||
int countHeight(bool plain = false);
|
|
||||||
|
|
||||||
void installedLocally(uint64 setId);
|
void installedLocally(uint64 setId);
|
||||||
void notInstalledLocally(uint64 setId);
|
void notInstalledLocally(uint64 setId);
|
||||||
|
@ -253,10 +284,12 @@ protected:
|
||||||
void mousePressEvent(QMouseEvent *e) override;
|
void mousePressEvent(QMouseEvent *e) override;
|
||||||
void mouseReleaseEvent(QMouseEvent *e) override;
|
void mouseReleaseEvent(QMouseEvent *e) override;
|
||||||
void mouseMoveEvent(QMouseEvent *e) override;
|
void mouseMoveEvent(QMouseEvent *e) override;
|
||||||
|
void resizeEvent(QResizeEvent *e) override;
|
||||||
void paintEvent(QPaintEvent *e) override;
|
void paintEvent(QPaintEvent *e) override;
|
||||||
void leaveEventHook(QEvent *e) override;
|
void leaveEventHook(QEvent *e) override;
|
||||||
void leaveToChildEvent(QEvent *e, QWidget *child) override;
|
void leaveToChildEvent(QEvent *e, QWidget *child) override;
|
||||||
void enterFromChildEvent(QEvent *e, QWidget *child) override;
|
void enterFromChildEvent(QEvent *e, QWidget *child) override;
|
||||||
|
int countHeight() override;
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void onSettings();
|
void onSettings();
|
||||||
|
@ -275,19 +308,32 @@ signals:
|
||||||
|
|
||||||
void refreshIcons(bool scrollAnimation);
|
void refreshIcons(bool scrollAnimation);
|
||||||
void emptyInlineRows();
|
void emptyInlineRows();
|
||||||
|
|
||||||
void switchToEmoji();
|
|
||||||
|
|
||||||
void scrollToY(int y);
|
|
||||||
void scrollUpdated();
|
void scrollUpdated();
|
||||||
void disableScroll(bool dis);
|
|
||||||
void needRefreshPanels();
|
|
||||||
|
|
||||||
void saveConfigDelayed(int32 delay);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static constexpr bool kRefreshIconsScrollAnimation = true;
|
enum class Section {
|
||||||
static constexpr bool kRefreshIconsNoAnimation = false;
|
Inlines,
|
||||||
|
Gifs,
|
||||||
|
Featured,
|
||||||
|
Stickers,
|
||||||
|
};
|
||||||
|
class Controller;
|
||||||
|
|
||||||
|
static constexpr auto kRefreshIconsScrollAnimation = true;
|
||||||
|
static constexpr auto kRefreshIconsNoAnimation = false;
|
||||||
|
|
||||||
|
struct SectionInfo {
|
||||||
|
int section = 0;
|
||||||
|
int count = 0;
|
||||||
|
int top = 0;
|
||||||
|
int rowsCount = 0;
|
||||||
|
int rowsTop = 0;
|
||||||
|
int rowsBottom = 0;
|
||||||
|
};
|
||||||
|
template <typename Callback>
|
||||||
|
bool enumerateSections(Callback callback) const;
|
||||||
|
SectionInfo sectionInfo(int section) const;
|
||||||
|
SectionInfo sectionInfoByOffset(int yOffset) const;
|
||||||
|
|
||||||
void updateSelected();
|
void updateSelected();
|
||||||
void setSelected(int newSelected, int newSelectedFeaturedSet, int newSelectedFeaturedSetAdd);
|
void setSelected(int newSelected, int newSelectedFeaturedSet, int newSelectedFeaturedSetAdd);
|
||||||
|
@ -317,8 +363,9 @@ private:
|
||||||
return (_section == Section::Inlines) || (_section == Section::Gifs);
|
return (_section == Section::Inlines) || (_section == Section::Gifs);
|
||||||
}
|
}
|
||||||
|
|
||||||
void paintInlineItems(Painter &p, const QRect &r);
|
void paintInlineItems(Painter &p, QRect clip);
|
||||||
void paintStickers(Painter &p, const QRect &r);
|
void paintFeaturedStickers(Painter &p, QRect clip);
|
||||||
|
void paintStickers(Painter &p, QRect clip);
|
||||||
void paintSticker(Painter &p, Set &set, int y, int index, bool selected, bool deleteSelected);
|
void paintSticker(Painter &p, Set &set, int y, int index, bool selected, bool deleteSelected);
|
||||||
bool featuredHasAddButton(int index) const;
|
bool featuredHasAddButton(int index) const;
|
||||||
int featuredContentWidth() const;
|
int featuredContentWidth() const;
|
||||||
|
@ -334,24 +381,13 @@ private:
|
||||||
|
|
||||||
void selectEmoji(EmojiPtr emoji);
|
void selectEmoji(EmojiPtr emoji);
|
||||||
int stickersLeft() const;
|
int stickersLeft() const;
|
||||||
QRect stickerRect(int tab, int sel);
|
QRect stickerRect(int section, int sel);
|
||||||
|
|
||||||
int32 _maxHeight;
|
|
||||||
|
|
||||||
int _visibleTop = 0;
|
|
||||||
int _visibleBottom = 0;
|
|
||||||
|
|
||||||
Sets _mySets;
|
Sets _mySets;
|
||||||
Sets _featuredSets;
|
Sets _featuredSets;
|
||||||
OrderedSet<uint64> _installedLocallySets;
|
OrderedSet<uint64> _installedLocallySets;
|
||||||
QList<bool> _custom;
|
QList<bool> _custom;
|
||||||
|
|
||||||
enum class Section {
|
|
||||||
Inlines,
|
|
||||||
Gifs,
|
|
||||||
Featured,
|
|
||||||
Stickers,
|
|
||||||
};
|
|
||||||
Section _section = Section::Stickers;
|
Section _section = Section::Stickers;
|
||||||
UserData *_inlineBot;
|
UserData *_inlineBot;
|
||||||
QString _inlineBotTitle;
|
QString _inlineBotTitle;
|
||||||
|
@ -387,7 +423,7 @@ private:
|
||||||
|
|
||||||
int validateExistingInlineRows(const InlineResults &results);
|
int validateExistingInlineRows(const InlineResults &results);
|
||||||
void selectInlineResult(int row, int column);
|
void selectInlineResult(int row, int column);
|
||||||
void removeRecentSticker(int tab, int index);
|
void removeRecentSticker(int section, int index);
|
||||||
|
|
||||||
int _selected = -1;
|
int _selected = -1;
|
||||||
int _pressed = -1;
|
int _pressed = -1;
|
||||||
|
@ -404,57 +440,6 @@ private:
|
||||||
|
|
||||||
QTimer _previewTimer;
|
QTimer _previewTimer;
|
||||||
bool _previewShown = false;
|
bool _previewShown = false;
|
||||||
};
|
|
||||||
|
|
||||||
class EmojiPanel : public TWidget {
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
public:
|
|
||||||
EmojiPanel(QWidget *parent, const QString &text, uint64 setId, bool special, int32 wantedY); // Stickers::NoneSetId if in emoji
|
|
||||||
void setText(const QString &text);
|
|
||||||
void setDeleteVisible(bool isVisible);
|
|
||||||
|
|
||||||
int wantedY() const {
|
|
||||||
return _wantedY;
|
|
||||||
}
|
|
||||||
void setWantedY(int32 y) {
|
|
||||||
_wantedY = y;
|
|
||||||
}
|
|
||||||
|
|
||||||
signals:
|
|
||||||
void deleteClicked(quint64 setId);
|
|
||||||
void mousePressed();
|
|
||||||
|
|
||||||
public slots:
|
|
||||||
void onDelete();
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void paintEvent(QPaintEvent *e) override;
|
|
||||||
void mousePressEvent(QMouseEvent *e) override;
|
|
||||||
|
|
||||||
private:
|
|
||||||
void updateText();
|
|
||||||
|
|
||||||
int32 _wantedY;
|
|
||||||
QString _text, _fullText;
|
|
||||||
uint64 _setId;
|
|
||||||
bool _special, _deleteVisible;
|
|
||||||
Ui::IconButton *_delete = nullptr;
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
class EmojiSwitchButton : public Ui::AbstractButton {
|
|
||||||
public:
|
|
||||||
EmojiSwitchButton(QWidget *parent, bool toStickers); // otherwise toEmoji
|
|
||||||
void updateText(const QString &inlineBotUsername = QString());
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void paintEvent(QPaintEvent *e) override;
|
|
||||||
|
|
||||||
private:
|
|
||||||
bool _toStickers = false;
|
|
||||||
QString _text;
|
|
||||||
int _textWidth = 0;
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -515,10 +500,7 @@ private slots:
|
||||||
void refreshSavedGifs();
|
void refreshSavedGifs();
|
||||||
|
|
||||||
void onWndActiveChanged();
|
void onWndActiveChanged();
|
||||||
|
void onScroll();
|
||||||
void onScrollEmoji();
|
|
||||||
void onScrollStickers();
|
|
||||||
void onSwitch();
|
|
||||||
|
|
||||||
void onDisplaySet(quint64 setId);
|
void onDisplaySet(quint64 setId);
|
||||||
void onInstallSet(quint64 setId);
|
void onInstallSet(quint64 setId);
|
||||||
|
@ -526,10 +508,9 @@ private slots:
|
||||||
void onDelayedHide();
|
void onDelayedHide();
|
||||||
|
|
||||||
void onRefreshIcons(bool scrollAnimation);
|
void onRefreshIcons(bool scrollAnimation);
|
||||||
void onRefreshPanels();
|
|
||||||
|
|
||||||
void onSaveConfig();
|
void onSaveConfig();
|
||||||
void onSaveConfigDelayed(int32 delay);
|
void onSaveConfigDelayed(int delay);
|
||||||
|
|
||||||
void onInlineRequest();
|
void onInlineRequest();
|
||||||
void onEmptyInlineRows();
|
void onEmptyInlineRows();
|
||||||
|
@ -543,11 +524,45 @@ signals:
|
||||||
void updateStickers();
|
void updateStickers();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool inlineResultsShown() const;
|
using TabType = EmojiPanTabType;
|
||||||
|
class Tab {
|
||||||
|
public:
|
||||||
|
static constexpr auto kCount = 3;
|
||||||
|
|
||||||
|
Tab(TabType type, object_ptr<internal::BasicPanInner> widget);
|
||||||
|
|
||||||
|
object_ptr<internal::BasicPanInner> takeWidget();
|
||||||
|
void returnWidget(object_ptr<internal::BasicPanInner> widget);
|
||||||
|
|
||||||
|
TabType type() const {
|
||||||
|
return _type;
|
||||||
|
}
|
||||||
|
gsl::not_null<internal::BasicPanInner*> widget() const {
|
||||||
|
return _weak;
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveScrollTop();
|
||||||
|
void saveScrollTop(int scrollTop) {
|
||||||
|
_scrollTop = scrollTop;
|
||||||
|
}
|
||||||
|
int getScrollTop() const {
|
||||||
|
return _scrollTop;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
TabType _type = TabType::Emoji;
|
||||||
|
object_ptr<internal::BasicPanInner> _widget = { nullptr };
|
||||||
|
QPointer<internal::BasicPanInner> _weak;
|
||||||
|
int _scrollTop = 0;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
int marginTop() const;
|
||||||
|
int marginBottom() const;
|
||||||
int countBottom() const;
|
int countBottom() const;
|
||||||
void moveByBottom();
|
void moveByBottom();
|
||||||
|
void paintSlideFrame(Painter &p, TimeMs ms);
|
||||||
void paintContent(Painter &p);
|
void paintContent(Painter &p);
|
||||||
void performSwitch();
|
|
||||||
|
|
||||||
style::margins innerPadding() const;
|
style::margins innerPadding() const;
|
||||||
|
|
||||||
|
@ -562,20 +577,23 @@ private:
|
||||||
// This one is allowed to be not rounded.
|
// This one is allowed to be not rounded.
|
||||||
QRect verticalRect() const;
|
QRect verticalRect() const;
|
||||||
|
|
||||||
QImage grabForPanelAnimation();
|
enum class GrabType {
|
||||||
|
Panel,
|
||||||
|
Slide,
|
||||||
|
};
|
||||||
|
QImage grabForComplexAnimation(GrabType type);
|
||||||
void startShowAnimation();
|
void startShowAnimation();
|
||||||
void startOpacityAnimation(bool hiding);
|
void startOpacityAnimation(bool hiding);
|
||||||
void prepareCache();
|
void prepareCache();
|
||||||
|
|
||||||
class Container;
|
|
||||||
void opacityAnimationCallback();
|
void opacityAnimationCallback();
|
||||||
|
|
||||||
void hideFinished();
|
void hideFinished();
|
||||||
void showStarted();
|
void showStarted();
|
||||||
|
|
||||||
bool preventAutoHide() const;
|
bool preventAutoHide() const;
|
||||||
void setActiveTab(DBIEmojiTab tab);
|
void setActiveSection(DBIEmojiSection section);
|
||||||
void setCurrentTabIcon(DBIEmojiTab tab);
|
void setCurrentSectionIcon(DBIEmojiSection section);
|
||||||
|
|
||||||
void paintStickerSettingsIcon(Painter &p) const;
|
void paintStickerSettingsIcon(Painter &p) const;
|
||||||
void paintFeaturedStickerSetsBadge(Painter &p, int iconLeft) const;
|
void paintFeaturedStickerSetsBadge(Painter &p, int iconLeft) const;
|
||||||
|
@ -591,18 +609,40 @@ private:
|
||||||
void updateSelected();
|
void updateSelected();
|
||||||
void updateIcons();
|
void updateIcons();
|
||||||
|
|
||||||
void prepareTab(int &left, int top, int _width, Ui::IconButton *tab, DBIEmojiTab value);
|
void prepareSection(int &left, int top, int _width, Ui::IconButton *sectionIcon, DBIEmojiSection section);
|
||||||
void updatePanelsPositions(const QVector<internal::EmojiPanel*> &panels, int st);
|
|
||||||
|
|
||||||
void showAll();
|
void showAll();
|
||||||
void hideAll();
|
void hideForSliding();
|
||||||
|
|
||||||
|
void setWidgetToScrollArea();
|
||||||
|
void createTabsSlider();
|
||||||
|
void switchTab();
|
||||||
|
gsl::not_null<Tab*> getTab(TabType type) {
|
||||||
|
return &_tabs[static_cast<int>(type)];
|
||||||
|
}
|
||||||
|
gsl::not_null<const Tab*> getTab(TabType type) const {
|
||||||
|
return &_tabs[static_cast<int>(type)];
|
||||||
|
}
|
||||||
|
gsl::not_null<Tab*> currentTab() {
|
||||||
|
return getTab(_currentTabType);
|
||||||
|
}
|
||||||
|
gsl::not_null<const Tab*> currentTab() const {
|
||||||
|
return getTab(_currentTabType);
|
||||||
|
}
|
||||||
|
gsl::not_null<internal::EmojiPanInner*> emoji() const {
|
||||||
|
return static_cast<internal::EmojiPanInner*>(getTab(TabType::Emoji)->widget().get());
|
||||||
|
}
|
||||||
|
gsl::not_null<internal::StickerPanInner*> stickers() const {
|
||||||
|
return static_cast<internal::StickerPanInner*>(getTab(TabType::Stickers)->widget().get());
|
||||||
|
}
|
||||||
|
gsl::not_null<internal::StickerPanInner*> gifs() const {
|
||||||
|
return static_cast<internal::StickerPanInner*>(getTab(TabType::Gifs)->widget().get());
|
||||||
|
}
|
||||||
|
|
||||||
int _minTop = 0;
|
int _minTop = 0;
|
||||||
int _minBottom = 0;
|
int _minBottom = 0;
|
||||||
int _contentMaxHeight = 0;
|
int _contentMaxHeight = 0;
|
||||||
int _contentHeight = 0;
|
int _contentHeight = 0;
|
||||||
int _contentHeightEmoji = 0;
|
|
||||||
int _contentHeightStickers = 0;
|
|
||||||
bool _horizontal = false;
|
bool _horizontal = false;
|
||||||
|
|
||||||
int _width = 0;
|
int _width = 0;
|
||||||
|
@ -613,10 +653,11 @@ private:
|
||||||
Animation _a_show;
|
Animation _a_show;
|
||||||
|
|
||||||
bool _hiding = false;
|
bool _hiding = false;
|
||||||
|
bool _hideAfterSlide = false;
|
||||||
QPixmap _cache;
|
QPixmap _cache;
|
||||||
Animation _a_opacity;
|
Animation _a_opacity;
|
||||||
QTimer _hideTimer;
|
QTimer _hideTimer;
|
||||||
bool _inPanelGrab = false;
|
bool _inComplrexGrab = false;
|
||||||
|
|
||||||
class SlideAnimation;
|
class SlideAnimation;
|
||||||
std::unique_ptr<SlideAnimation> _slideAnimation;
|
std::unique_ptr<SlideAnimation> _slideAnimation;
|
||||||
|
@ -646,17 +687,12 @@ private:
|
||||||
anim::value _iconSelX;
|
anim::value _iconSelX;
|
||||||
TimeMs _iconsStartAnim = 0;
|
TimeMs _iconsStartAnim = 0;
|
||||||
|
|
||||||
bool _emojiShown = true;
|
object_ptr<Ui::SettingsSlider> _tabsSlider = { nullptr };
|
||||||
bool _shownFromInlineQuery = false;
|
object_ptr<Ui::PlainShadow> _topShadow;
|
||||||
|
object_ptr<Ui::PlainShadow> _bottomShadow;
|
||||||
object_ptr<Ui::ScrollArea> e_scroll;
|
object_ptr<Ui::ScrollArea> _scroll;
|
||||||
QPointer<internal::EmojiPanInner> e_inner;
|
std::array<Tab, Tab::kCount> _tabs;
|
||||||
QVector<internal::EmojiPanel*> e_panels;
|
TabType _currentTabType = TabType::Emoji;
|
||||||
object_ptr<internal::EmojiSwitchButton> e_switch;
|
|
||||||
object_ptr<Ui::ScrollArea> s_scroll;
|
|
||||||
QPointer<internal::StickerPanInner> s_inner;
|
|
||||||
QVector<internal::EmojiPanel*> s_panels;
|
|
||||||
object_ptr<internal::EmojiSwitchButton> s_switch;
|
|
||||||
|
|
||||||
uint64 _displayingSetId = 0;
|
uint64 _displayingSetId = 0;
|
||||||
uint64 _removingSetId = 0;
|
uint64 _removingSetId = 0;
|
||||||
|
@ -669,7 +705,6 @@ private:
|
||||||
|
|
||||||
void inlineBotChanged();
|
void inlineBotChanged();
|
||||||
int32 showInlineRows(bool newResults);
|
int32 showInlineRows(bool newResults);
|
||||||
void recountContentMaxHeight();
|
|
||||||
bool refreshInlineRows(int32 *added = 0);
|
bool refreshInlineRows(int32 *added = 0);
|
||||||
UserData *_inlineBot = nullptr;
|
UserData *_inlineBot = nullptr;
|
||||||
PeerData *_inlineQueryPeer = nullptr;
|
PeerData *_inlineQueryPeer = nullptr;
|
||||||
|
|
|
@ -24,6 +24,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
|
|
||||||
namespace Stickers {
|
namespace Stickers {
|
||||||
|
|
||||||
|
constexpr auto kPanPerRow = 5;
|
||||||
|
|
||||||
void applyArchivedResult(const MTPDmessages_stickerSetInstallResultArchive &d);
|
void applyArchivedResult(const MTPDmessages_stickerSetInstallResultArchive &d);
|
||||||
bool applyArchivedResultFake(); // For testing.
|
bool applyArchivedResultFake(); // For testing.
|
||||||
void installLocally(uint64 setId);
|
void installLocally(uint64 setId);
|
||||||
|
|
|
@ -109,9 +109,11 @@ stickersSettingsUnreadPosition: point(4px, 5px);
|
||||||
|
|
||||||
emojiPanMargins: margins(10px, 10px, 10px, 10px);
|
emojiPanMargins: margins(10px, 10px, 10px, 10px);
|
||||||
|
|
||||||
emojiScroll: ScrollArea(defaultSolidScroll) {
|
|
||||||
deltat: 48px;
|
emojiTabs: SettingsSlider(defaultTabsSlider) {
|
||||||
|
rippleRoundRadius: buttonRadius;
|
||||||
}
|
}
|
||||||
|
emojiScroll: defaultSolidScroll;
|
||||||
emojiRecent: icon {{ "emoji_recent", emojiIconFg }};
|
emojiRecent: icon {{ "emoji_recent", emojiIconFg }};
|
||||||
emojiRecentActive: icon {{ "emoji_recent", emojiIconFgActive }};
|
emojiRecentActive: icon {{ "emoji_recent", emojiIconFgActive }};
|
||||||
emojiPeople: icon {{ "emoji_people", emojiIconFg }};
|
emojiPeople: icon {{ "emoji_people", emojiIconFg }};
|
||||||
|
@ -128,8 +130,6 @@ emojiObjects: icon {{ "emoji_objects", emojiIconFg }};
|
||||||
emojiObjectsActive: icon {{ "emoji_objects", emojiIconFgActive }};
|
emojiObjectsActive: icon {{ "emoji_objects", emojiIconFgActive }};
|
||||||
emojiSymbols: icon {{ "emoji_symbols", emojiIconFg }};
|
emojiSymbols: icon {{ "emoji_symbols", emojiIconFg }};
|
||||||
emojiSymbolsActive: icon {{ "emoji_symbols", emojiIconFgActive }};
|
emojiSymbolsActive: icon {{ "emoji_symbols", emojiIconFgActive }};
|
||||||
emojiSavedGifs: icon {{ "emoji_gif", emojiIconFg }};
|
|
||||||
emojiSavedGifsActive: icon {{ "emoji_gif", emojiIconFgActive }};
|
|
||||||
|
|
||||||
emojiCategory: IconButton {
|
emojiCategory: IconButton {
|
||||||
width: 42px;
|
width: 42px;
|
||||||
|
@ -152,12 +152,11 @@ emojiPanAnimation: PanelAnimation(defaultPanelAnimation) {
|
||||||
emojiPanPadding: 12px;
|
emojiPanPadding: 12px;
|
||||||
emojiPanSize: size(45px, 41px);
|
emojiPanSize: size(45px, 41px);
|
||||||
emojiPanWidth: 345px;
|
emojiPanWidth: 345px;
|
||||||
emojiPanMinHeight: 206px;
|
|
||||||
emojiPanMaxHeight: 366px;
|
emojiPanMaxHeight: 366px;
|
||||||
emojiPanShowDuration: 200;
|
emojiPanShowDuration: 2000;
|
||||||
emojiPanDuration: 200;
|
emojiPanDuration: 2000;
|
||||||
emojiPanHover: windowBgOver;
|
emojiPanHover: windowBgOver;
|
||||||
emojiPanSlideDuration: 200;
|
emojiPanSlideDuration: 2000;
|
||||||
|
|
||||||
emojiPanHeader: 42px;
|
emojiPanHeader: 42px;
|
||||||
emojiPanHeaderFont: semiboldFont;
|
emojiPanHeaderFont: semiboldFont;
|
||||||
|
|
|
@ -48,8 +48,6 @@ using FileKey = quint64;
|
||||||
constexpr char tdfMagic[] = { 'T', 'D', 'F', '$' };
|
constexpr char tdfMagic[] = { 'T', 'D', 'F', '$' };
|
||||||
constexpr int tdfMagicLen = sizeof(tdfMagic);
|
constexpr int tdfMagicLen = sizeof(tdfMagic);
|
||||||
|
|
||||||
bool _cacheLastSeenWarningSeen = false;
|
|
||||||
|
|
||||||
QString toFilePart(FileKey val) {
|
QString toFilePart(FileKey val) {
|
||||||
QString result;
|
QString result;
|
||||||
result.reserve(0x10);
|
result.reserve(0x10);
|
||||||
|
@ -556,7 +554,7 @@ enum {
|
||||||
dbiDownloadPath = 0x33,
|
dbiDownloadPath = 0x33,
|
||||||
dbiAutoDownload = 0x34,
|
dbiAutoDownload = 0x34,
|
||||||
dbiSavedGifsLimit = 0x35,
|
dbiSavedGifsLimit = 0x35,
|
||||||
dbiShowingSavedGifs = 0x36,
|
dbiShowingSavedGifsOld = 0x36,
|
||||||
dbiAutoPlay = 0x37,
|
dbiAutoPlay = 0x37,
|
||||||
dbiAdaptiveForWide = 0x38,
|
dbiAdaptiveForWide = 0x38,
|
||||||
dbiHiddenPinnedMessages = 0x39,
|
dbiHiddenPinnedMessages = 0x39,
|
||||||
|
@ -574,7 +572,8 @@ enum {
|
||||||
dbiUseExternalVideoPlayer = 0x49,
|
dbiUseExternalVideoPlayer = 0x49,
|
||||||
dbiDcOptions = 0x4a,
|
dbiDcOptions = 0x4a,
|
||||||
dbiMtpAuthorization = 0x4b,
|
dbiMtpAuthorization = 0x4b,
|
||||||
dbiLastSeenWarningSeen = 0x4c,
|
dbiLastSeenWarningSeenOld = 0x4c,
|
||||||
|
dbiAuthSessionData = 0x4d,
|
||||||
|
|
||||||
dbiEncryptedWithSalt = 333,
|
dbiEncryptedWithSalt = 333,
|
||||||
dbiEncrypted = 444,
|
dbiEncrypted = 444,
|
||||||
|
@ -633,17 +632,25 @@ int32 _storageImagesSize = 0, _storageStickersSize = 0, _storageAudiosSize = 0;
|
||||||
bool _mapChanged = false;
|
bool _mapChanged = false;
|
||||||
int32 _oldMapVersion = 0, _oldSettingsVersion = 0;
|
int32 _oldMapVersion = 0, _oldSettingsVersion = 0;
|
||||||
|
|
||||||
enum WriteMapWhen {
|
enum class WriteMapWhen {
|
||||||
WriteMapNow,
|
Now,
|
||||||
WriteMapFast,
|
Fast,
|
||||||
WriteMapSoon,
|
Soon,
|
||||||
};
|
};
|
||||||
|
|
||||||
void _writeMap(WriteMapWhen when = WriteMapSoon);
|
std::unique_ptr<AuthSessionData> AuthSessionDataCache;
|
||||||
|
AuthSessionData &GetAuthSessionDataCache() {
|
||||||
|
if (!AuthSessionDataCache) {
|
||||||
|
AuthSessionDataCache = std::make_unique<AuthSessionData>();
|
||||||
|
}
|
||||||
|
return *AuthSessionDataCache;
|
||||||
|
}
|
||||||
|
|
||||||
void _writeLocations(WriteMapWhen when = WriteMapSoon) {
|
void _writeMap(WriteMapWhen when = WriteMapWhen::Soon);
|
||||||
if (when != WriteMapNow) {
|
|
||||||
_manager->writeLocations(when == WriteMapFast);
|
void _writeLocations(WriteMapWhen when = WriteMapWhen::Soon) {
|
||||||
|
if (when != WriteMapWhen::Now) {
|
||||||
|
_manager->writeLocations(when == WriteMapWhen::Fast);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!_working()) return;
|
if (!_working()) return;
|
||||||
|
@ -660,7 +667,7 @@ void _writeLocations(WriteMapWhen when = WriteMapSoon) {
|
||||||
if (!_locationsKey) {
|
if (!_locationsKey) {
|
||||||
_locationsKey = genKey();
|
_locationsKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
quint32 size = 0;
|
quint32 size = 0;
|
||||||
for (FileLocations::const_iterator i = _fileLocations.cbegin(), e = _fileLocations.cend(); i != e; ++i) {
|
for (FileLocations::const_iterator i = _fileLocations.cbegin(), e = _fileLocations.cend(); i != e; ++i) {
|
||||||
|
@ -798,7 +805,7 @@ void _writeReportSpamStatuses() {
|
||||||
if (!_reportSpamStatusesKey) {
|
if (!_reportSpamStatusesKey) {
|
||||||
_reportSpamStatusesKey = genKey();
|
_reportSpamStatusesKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
const ReportSpamStatuses &statuses(cReportSpamStatuses());
|
const ReportSpamStatuses &statuses(cReportSpamStatuses());
|
||||||
|
|
||||||
|
@ -920,7 +927,6 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting
|
||||||
Messenger::Instance().setMtpMainDcId(dcId);
|
Messenger::Instance().setMtpMainDcId(dcId);
|
||||||
if (userId) {
|
if (userId) {
|
||||||
Messenger::Instance().authSessionCreate(UserId(userId));
|
Messenger::Instance().authSessionCreate(UserId(userId));
|
||||||
AuthSession::Current().data().setLastSeenWarningSeen(_cacheLastSeenWarningSeen);
|
|
||||||
}
|
}
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
|
@ -940,9 +946,6 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting
|
||||||
if (!_checkStreamStatus(stream)) return false;
|
if (!_checkStreamStatus(stream)) return false;
|
||||||
|
|
||||||
Messenger::Instance().setMtpAuthorization(serialized);
|
Messenger::Instance().setMtpAuthorization(serialized);
|
||||||
if (AuthSession::Exists()) {
|
|
||||||
AuthSession::Current().data().setLastSeenWarningSeen(_cacheLastSeenWarningSeen);
|
|
||||||
}
|
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbiAutoStart: {
|
case dbiAutoStart: {
|
||||||
|
@ -1035,12 +1038,10 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting
|
||||||
Global::SetIncludeMuted(v == 1);
|
Global::SetIncludeMuted(v == 1);
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbiShowingSavedGifs: {
|
case dbiShowingSavedGifsOld: {
|
||||||
qint32 v;
|
qint32 v;
|
||||||
stream >> v;
|
stream >> v;
|
||||||
if (!_checkStreamStatus(stream)) return false;
|
if (!_checkStreamStatus(stream)) return false;
|
||||||
|
|
||||||
cSetShowingSavedGifs(v == 1);
|
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbiDesktopNotify: {
|
case dbiDesktopNotify: {
|
||||||
|
@ -1090,16 +1091,20 @@ bool _readSetting(quint32 blockId, QDataStream &stream, int version, ReadSetting
|
||||||
Global::SetDialogsWidthRatio(v / 1000000.);
|
Global::SetDialogsWidthRatio(v / 1000000.);
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbiLastSeenWarningSeen: {
|
case dbiLastSeenWarningSeenOld: {
|
||||||
qint32 v;
|
qint32 v;
|
||||||
stream >> v;
|
stream >> v;
|
||||||
if (!_checkStreamStatus(stream)) return false;
|
if (!_checkStreamStatus(stream)) return false;
|
||||||
|
|
||||||
if (AuthSession::Exists()) {
|
GetAuthSessionDataCache().setLastSeenWarningSeen(v == 1);
|
||||||
AuthSession::Current().data().setLastSeenWarningSeen(v == 1);
|
} break;
|
||||||
} else {
|
|
||||||
_cacheLastSeenWarningSeen = (v == 1);
|
case dbiAuthSessionData: {
|
||||||
}
|
QByteArray v;
|
||||||
|
stream >> v;
|
||||||
|
if (!_checkStreamStatus(stream)) return false;
|
||||||
|
|
||||||
|
GetAuthSessionDataCache().constructFromSerialized(v);
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbiWorkMode: {
|
case dbiWorkMode: {
|
||||||
|
@ -1694,16 +1699,18 @@ void _writeUserSettings() {
|
||||||
if (!_userSettingsKey) {
|
if (!_userSettingsKey) {
|
||||||
_userSettingsKey = genKey();
|
_userSettingsKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto recentEmojiPreloadData = cRecentEmojiPreload();
|
auto recentEmojiPreloadData = cRecentEmojiPreload();
|
||||||
if (recentEmojiPreloadData.isEmpty()) {
|
if (recentEmojiPreloadData.isEmpty()) {
|
||||||
recentEmojiPreloadData.reserve(cGetRecentEmoji().size());
|
recentEmojiPreloadData.reserve(Ui::Emoji::GetRecent().size());
|
||||||
for (auto &item : cGetRecentEmoji()) {
|
for (auto &item : Ui::Emoji::GetRecent()) {
|
||||||
recentEmojiPreloadData.push_back(qMakePair(item.first->id(), item.second));
|
recentEmojiPreloadData.push_back(qMakePair(item.first->id(), item.second));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
auto userDataInstance = AuthSessionDataCache ? AuthSessionDataCache.get() : AuthSession::Exists() ? &AuthSession::Current().data() : nullptr;
|
||||||
|
auto userData = userDataInstance ? userDataInstance->serialize() : QByteArray();
|
||||||
|
|
||||||
uint32 size = 21 * (sizeof(quint32) + sizeof(qint32));
|
uint32 size = 21 * (sizeof(quint32) + sizeof(qint32));
|
||||||
size += sizeof(quint32) + Serialize::stringSize(Global::AskDownloadPath() ? QString() : Global::DownloadPath()) + Serialize::bytearraySize(Global::AskDownloadPath() ? QByteArray() : Global::DownloadPathBookmark());
|
size += sizeof(quint32) + Serialize::stringSize(Global::AskDownloadPath() ? QString() : Global::DownloadPath()) + Serialize::bytearraySize(Global::AskDownloadPath() ? QByteArray() : Global::DownloadPathBookmark());
|
||||||
|
@ -1721,6 +1728,9 @@ void _writeUserSettings() {
|
||||||
if (!Global::HiddenPinnedMessages().isEmpty()) {
|
if (!Global::HiddenPinnedMessages().isEmpty()) {
|
||||||
size += sizeof(quint32) + sizeof(qint32) + Global::HiddenPinnedMessages().size() * (sizeof(PeerId) + sizeof(MsgId));
|
size += sizeof(quint32) + sizeof(qint32) + Global::HiddenPinnedMessages().size() * (sizeof(PeerId) + sizeof(MsgId));
|
||||||
}
|
}
|
||||||
|
if (!userData.isEmpty()) {
|
||||||
|
size += sizeof(quint32) + Serialize::bytearraySize(userData);
|
||||||
|
}
|
||||||
|
|
||||||
EncryptedDescriptor data(size);
|
EncryptedDescriptor data(size);
|
||||||
data.stream << quint32(dbiSendKey) << qint32(cCtrlEnter() ? dbiskCtrlEnter : dbiskEnter);
|
data.stream << quint32(dbiSendKey) << qint32(cCtrlEnter() ? dbiskCtrlEnter : dbiskEnter);
|
||||||
|
@ -1730,7 +1740,6 @@ void _writeUserSettings() {
|
||||||
data.stream << quint32(dbiReplaceEmojis) << qint32(cReplaceEmojis() ? 1 : 0);
|
data.stream << quint32(dbiReplaceEmojis) << qint32(cReplaceEmojis() ? 1 : 0);
|
||||||
data.stream << quint32(dbiSoundNotify) << qint32(Global::SoundNotify());
|
data.stream << quint32(dbiSoundNotify) << qint32(Global::SoundNotify());
|
||||||
data.stream << quint32(dbiIncludeMuted) << qint32(Global::IncludeMuted());
|
data.stream << quint32(dbiIncludeMuted) << qint32(Global::IncludeMuted());
|
||||||
data.stream << quint32(dbiShowingSavedGifs) << qint32(cShowingSavedGifs());
|
|
||||||
data.stream << quint32(dbiDesktopNotify) << qint32(Global::DesktopNotify());
|
data.stream << quint32(dbiDesktopNotify) << qint32(Global::DesktopNotify());
|
||||||
data.stream << quint32(dbiNotifyView) << qint32(Global::NotifyView());
|
data.stream << quint32(dbiNotifyView) << qint32(Global::NotifyView());
|
||||||
data.stream << quint32(dbiNativeNotifications) << qint32(Global::NativeNotifications());
|
data.stream << quint32(dbiNativeNotifications) << qint32(Global::NativeNotifications());
|
||||||
|
@ -1747,9 +1756,10 @@ void _writeUserSettings() {
|
||||||
data.stream << quint32(dbiModerateMode) << qint32(Global::ModerateModeEnabled() ? 1 : 0);
|
data.stream << quint32(dbiModerateMode) << qint32(Global::ModerateModeEnabled() ? 1 : 0);
|
||||||
data.stream << quint32(dbiAutoPlay) << qint32(cAutoPlayGif() ? 1 : 0);
|
data.stream << quint32(dbiAutoPlay) << qint32(cAutoPlayGif() ? 1 : 0);
|
||||||
data.stream << quint32(dbiDialogsWidthRatio) << qint32(snap(qRound(Global::DialogsWidthRatio() * 1000000), 0, 1000000));
|
data.stream << quint32(dbiDialogsWidthRatio) << qint32(snap(qRound(Global::DialogsWidthRatio() * 1000000), 0, 1000000));
|
||||||
auto lastSeenWarningSeen = (AuthSession::Exists() ? AuthSession::Current().data().lastSeenWarningSeen() : _cacheLastSeenWarningSeen);
|
|
||||||
data.stream << quint32(dbiLastSeenWarningSeen) << qint32(lastSeenWarningSeen ? 1 : 0);
|
|
||||||
data.stream << quint32(dbiUseExternalVideoPlayer) << qint32(cUseExternalVideoPlayer());
|
data.stream << quint32(dbiUseExternalVideoPlayer) << qint32(cUseExternalVideoPlayer());
|
||||||
|
if (!userData.isEmpty()) {
|
||||||
|
data.stream << quint32(dbiAuthSessionData) << userData;
|
||||||
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
data.stream << quint32(dbiRecentEmoji) << recentEmojiPreloadData;
|
data.stream << quint32(dbiRecentEmoji) << recentEmojiPreloadData;
|
||||||
|
@ -2055,6 +2065,13 @@ ReadMapState _readMap(const QByteArray &pass) {
|
||||||
_readUserSettings();
|
_readUserSettings();
|
||||||
_readMtpData();
|
_readMtpData();
|
||||||
|
|
||||||
|
if (AuthSessionDataCache) {
|
||||||
|
if (AuthSession::Exists()) {
|
||||||
|
AuthSession::Current().data().copyFrom(*AuthSessionDataCache);
|
||||||
|
}
|
||||||
|
AuthSessionDataCache.reset();
|
||||||
|
}
|
||||||
|
|
||||||
LOG(("Map read time: %1").arg(getms() - ms));
|
LOG(("Map read time: %1").arg(getms() - ms));
|
||||||
if (_oldSettingsVersion < AppVersion) {
|
if (_oldSettingsVersion < AppVersion) {
|
||||||
writeSettings();
|
writeSettings();
|
||||||
|
@ -2063,8 +2080,8 @@ ReadMapState _readMap(const QByteArray &pass) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void _writeMap(WriteMapWhen when) {
|
void _writeMap(WriteMapWhen when) {
|
||||||
if (when != WriteMapNow) {
|
if (when != WriteMapWhen::Now) {
|
||||||
_manager->writeMap(when == WriteMapFast);
|
_manager->writeMap(when == WriteMapWhen::Fast);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_manager->writingMap();
|
_manager->writingMap();
|
||||||
|
@ -2183,7 +2200,7 @@ void _writeMap(WriteMapWhen when) {
|
||||||
|
|
||||||
void finish() {
|
void finish() {
|
||||||
if (_manager) {
|
if (_manager) {
|
||||||
_writeMap(WriteMapNow);
|
_writeMap(WriteMapWhen::Now);
|
||||||
_manager->finish();
|
_manager->finish();
|
||||||
_manager->deleteLater();
|
_manager->deleteLater();
|
||||||
_manager = 0;
|
_manager = 0;
|
||||||
|
@ -2350,9 +2367,9 @@ void reset() {
|
||||||
_savedGifsKey = 0;
|
_savedGifsKey = 0;
|
||||||
_backgroundKey = _userSettingsKey = _recentHashtagsAndBotsKey = _savedPeersKey = 0;
|
_backgroundKey = _userSettingsKey = _recentHashtagsAndBotsKey = _savedPeersKey = 0;
|
||||||
_oldMapVersion = _oldSettingsVersion = 0;
|
_oldMapVersion = _oldSettingsVersion = 0;
|
||||||
_cacheLastSeenWarningSeen = false;
|
AuthSessionDataCache.reset();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapNow);
|
_writeMap(WriteMapWhen::Now);
|
||||||
|
|
||||||
_writeMtpData();
|
_writeMtpData();
|
||||||
}
|
}
|
||||||
|
@ -2371,7 +2388,7 @@ void setPasscode(const QByteArray &passcode) {
|
||||||
_passKeyEncrypted = FileWriteDescriptor::prepareEncrypted(passKeyData, PassKey);
|
_passKeyEncrypted = FileWriteDescriptor::prepareEncrypted(passKeyData, PassKey);
|
||||||
|
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapNow);
|
_writeMap(WriteMapWhen::Now);
|
||||||
|
|
||||||
Global::SetLocalPasscode(!passcode.isEmpty());
|
Global::SetLocalPasscode(!passcode.isEmpty());
|
||||||
Global::RefLocalPasscodeChanged().notify();
|
Global::RefLocalPasscodeChanged().notify();
|
||||||
|
@ -2381,7 +2398,7 @@ ReadMapState readMap(const QByteArray &pass) {
|
||||||
ReadMapState result = _readMap(pass);
|
ReadMapState result = _readMap(pass);
|
||||||
if (result == ReadMapFailed) {
|
if (result == ReadMapFailed) {
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapNow);
|
_writeMap(WriteMapWhen::Now);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -2412,7 +2429,7 @@ void writeDrafts(const PeerId &peer, const MessageDraft &localDraft, const Messa
|
||||||
if (i == _draftsMap.cend()) {
|
if (i == _draftsMap.cend()) {
|
||||||
i = _draftsMap.insert(peer, genKey());
|
i = _draftsMap.insert(peer, genKey());
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto msgTags = Ui::FlatTextarea::serializeTagsList(localDraft.textWithTags.tags);
|
auto msgTags = Ui::FlatTextarea::serializeTagsList(localDraft.textWithTags.tags);
|
||||||
|
@ -2552,7 +2569,7 @@ void writeDraftCursors(const PeerId &peer, const MessageCursor &msgCursor, const
|
||||||
if (i == _draftCursorsMap.cend()) {
|
if (i == _draftCursorsMap.cend()) {
|
||||||
i = _draftCursorsMap.insert(peer, genKey());
|
i = _draftCursorsMap.insert(peer, genKey());
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
|
|
||||||
EncryptedDescriptor data(sizeof(quint64) + sizeof(qint32) * 3);
|
EncryptedDescriptor data(sizeof(quint64) + sizeof(qint32) * 3);
|
||||||
|
@ -2585,7 +2602,7 @@ void writeFileLocation(MediaKey location, const FileLocation &local) {
|
||||||
if (i.value().second == local) {
|
if (i.value().second == local) {
|
||||||
if (i.value().first != location) {
|
if (i.value().first != location) {
|
||||||
_fileLocationAliases.insert(location, i.value().first);
|
_fileLocationAliases.insert(location, i.value().first);
|
||||||
_writeLocations(WriteMapFast);
|
_writeLocations(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -2601,7 +2618,7 @@ void writeFileLocation(MediaKey location, const FileLocation &local) {
|
||||||
}
|
}
|
||||||
_fileLocations.insert(location, local);
|
_fileLocations.insert(location, local);
|
||||||
_fileLocationPairs.insert(local.fname, FileLocationPair(location, local));
|
_fileLocationPairs.insert(local.fname, FileLocationPair(location, local));
|
||||||
_writeLocations(WriteMapFast);
|
_writeLocations(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
|
|
||||||
FileLocation readFileLocation(MediaKey location, bool check) {
|
FileLocation readFileLocation(MediaKey location, bool check) {
|
||||||
|
@ -3192,7 +3209,7 @@ void _writeStickerSets(FileKey &stickersKey, CheckSet checkSet, const Stickers::
|
||||||
if (!stickersKey) {
|
if (!stickersKey) {
|
||||||
stickersKey = genKey();
|
stickersKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
EncryptedDescriptor data(size);
|
EncryptedDescriptor data(size);
|
||||||
data.stream << quint32(setsCount) << hashToWrite;
|
data.stream << quint32(setsCount) << hashToWrite;
|
||||||
|
@ -3610,7 +3627,7 @@ void writeSavedGifs() {
|
||||||
if (!_savedGifsKey) {
|
if (!_savedGifsKey) {
|
||||||
_savedGifsKey = genKey();
|
_savedGifsKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
EncryptedDescriptor data(size);
|
EncryptedDescriptor data(size);
|
||||||
data.stream << quint32(saved.size());
|
data.stream << quint32(saved.size());
|
||||||
|
@ -3667,7 +3684,7 @@ void writeBackground(int32 id, const QImage &img) {
|
||||||
if (!_backgroundKey) {
|
if (!_backgroundKey) {
|
||||||
_backgroundKey = genKey();
|
_backgroundKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
quint32 size = sizeof(qint32) + sizeof(quint32) + (bmp.isEmpty() ? 0 : (sizeof(quint32) + bmp.size()));
|
quint32 size = sizeof(qint32) + sizeof(quint32) + (bmp.isEmpty() ? 0 : (sizeof(quint32) + bmp.size()));
|
||||||
EncryptedDescriptor data(size);
|
EncryptedDescriptor data(size);
|
||||||
|
@ -4028,7 +4045,7 @@ void writeRecentHashtagsAndBots() {
|
||||||
if (!_recentHashtagsAndBotsKey) {
|
if (!_recentHashtagsAndBotsKey) {
|
||||||
_recentHashtagsAndBotsKey = genKey();
|
_recentHashtagsAndBotsKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
quint32 size = sizeof(quint32) * 3, writeCnt = 0, searchCnt = 0, botsCnt = cRecentInlineBots().size();
|
quint32 size = sizeof(quint32) * 3, writeCnt = 0, searchCnt = 0, botsCnt = cRecentInlineBots().size();
|
||||||
for (RecentHashtagPack::const_iterator i = write.cbegin(), e = write.cend(); i != e; ++i) {
|
for (RecentHashtagPack::const_iterator i = write.cbegin(), e = write.cend(); i != e; ++i) {
|
||||||
|
@ -4133,7 +4150,7 @@ void writeSavedPeers() {
|
||||||
if (!_savedPeersKey) {
|
if (!_savedPeersKey) {
|
||||||
_savedPeersKey = genKey();
|
_savedPeersKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
quint32 size = sizeof(quint32);
|
quint32 size = sizeof(quint32);
|
||||||
for (SavedPeers::const_iterator i = saved.cbegin(); i != saved.cend(); ++i) {
|
for (SavedPeers::const_iterator i = saved.cbegin(); i != saved.cend(); ++i) {
|
||||||
|
@ -4234,7 +4251,7 @@ void writeTrustedBots() {
|
||||||
if (!_trustedBotsKey) {
|
if (!_trustedBotsKey) {
|
||||||
_trustedBotsKey = genKey();
|
_trustedBotsKey = genKey();
|
||||||
_mapChanged = true;
|
_mapChanged = true;
|
||||||
_writeMap(WriteMapFast);
|
_writeMap(WriteMapWhen::Fast);
|
||||||
}
|
}
|
||||||
quint32 size = sizeof(qint32) + _trustedBots.size() * sizeof(quint64);
|
quint32 size = sizeof(qint32) + _trustedBots.size() * sizeof(quint64);
|
||||||
EncryptedDescriptor data(size);
|
EncryptedDescriptor data(size);
|
||||||
|
@ -4586,11 +4603,11 @@ void Manager::writingLocations() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::mapWriteTimeout() {
|
void Manager::mapWriteTimeout() {
|
||||||
_writeMap(WriteMapNow);
|
_writeMap(WriteMapWhen::Now);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::locationsWriteTimeout() {
|
void Manager::locationsWriteTimeout() {
|
||||||
_writeLocations(WriteMapNow);
|
_writeLocations(WriteMapWhen::Now);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Manager::finish() {
|
void Manager::finish() {
|
||||||
|
|
|
@ -15239,23 +15239,23 @@ void Init() {
|
||||||
Items.emplace_back(internal::ComputeId(0xd83c, 0xddff, 0xd83c, 0xddfc), 6, 54, false, false, nullptr, tag);
|
Items.emplace_back(internal::ComputeId(0xd83c, 0xddff, 0xd83c, 0xddfc), 6, 54, false, false, nullptr, tag);
|
||||||
}
|
}
|
||||||
|
|
||||||
int GetPackCount(DBIEmojiTab tab) {
|
int GetPackCount(DBIEmojiSection tab) {
|
||||||
switch (tab) {
|
switch (tab) {
|
||||||
case dbietPeople: return 291;
|
case dbiesPeople: return 291;
|
||||||
case dbietNature: return 159;
|
case dbiesNature: return 159;
|
||||||
case dbietFood: return 86;
|
case dbiesFood: return 86;
|
||||||
case dbietActivity: return 80;
|
case dbiesActivity: return 80;
|
||||||
case dbietTravel: return 119;
|
case dbiesTravel: return 119;
|
||||||
case dbietObjects: return 173;
|
case dbiesObjects: return 173;
|
||||||
case dbietSymbols: return 524;
|
case dbiesSymbols: return 524;
|
||||||
case dbietRecent: return cGetRecentEmoji().size();
|
case dbiesRecent: return GetRecent().size();
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
EmojiPack GetPack(DBIEmojiTab tab) {
|
EmojiPack GetPack(DBIEmojiSection tab) {
|
||||||
switch (tab) {
|
switch (tab) {
|
||||||
case dbietPeople: {
|
case dbiesPeople: {
|
||||||
static auto result = EmojiPack();
|
static auto result = EmojiPack();
|
||||||
if (result.isEmpty()) {
|
if (result.isEmpty()) {
|
||||||
result.reserve(291);
|
result.reserve(291);
|
||||||
|
@ -15554,7 +15554,7 @@ EmojiPack GetPack(DBIEmojiTab tab) {
|
||||||
return result;
|
return result;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbietNature: {
|
case dbiesNature: {
|
||||||
static auto result = EmojiPack();
|
static auto result = EmojiPack();
|
||||||
if (result.isEmpty()) {
|
if (result.isEmpty()) {
|
||||||
result.reserve(159);
|
result.reserve(159);
|
||||||
|
@ -15721,7 +15721,7 @@ EmojiPack GetPack(DBIEmojiTab tab) {
|
||||||
return result;
|
return result;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbietFood: {
|
case dbiesFood: {
|
||||||
static auto result = EmojiPack();
|
static auto result = EmojiPack();
|
||||||
if (result.isEmpty()) {
|
if (result.isEmpty()) {
|
||||||
result.reserve(86);
|
result.reserve(86);
|
||||||
|
@ -15815,7 +15815,7 @@ EmojiPack GetPack(DBIEmojiTab tab) {
|
||||||
return result;
|
return result;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbietActivity: {
|
case dbiesActivity: {
|
||||||
static auto result = EmojiPack();
|
static auto result = EmojiPack();
|
||||||
if (result.isEmpty()) {
|
if (result.isEmpty()) {
|
||||||
result.reserve(80);
|
result.reserve(80);
|
||||||
|
@ -15903,7 +15903,7 @@ EmojiPack GetPack(DBIEmojiTab tab) {
|
||||||
return result;
|
return result;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbietTravel: {
|
case dbiesTravel: {
|
||||||
static auto result = EmojiPack();
|
static auto result = EmojiPack();
|
||||||
if (result.isEmpty()) {
|
if (result.isEmpty()) {
|
||||||
result.reserve(119);
|
result.reserve(119);
|
||||||
|
@ -16030,7 +16030,7 @@ EmojiPack GetPack(DBIEmojiTab tab) {
|
||||||
return result;
|
return result;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbietObjects: {
|
case dbiesObjects: {
|
||||||
static auto result = EmojiPack();
|
static auto result = EmojiPack();
|
||||||
if (result.isEmpty()) {
|
if (result.isEmpty()) {
|
||||||
result.reserve(173);
|
result.reserve(173);
|
||||||
|
@ -16211,7 +16211,7 @@ EmojiPack GetPack(DBIEmojiTab tab) {
|
||||||
return result;
|
return result;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbietSymbols: {
|
case dbiesSymbols: {
|
||||||
static auto result = EmojiPack();
|
static auto result = EmojiPack();
|
||||||
if (result.isEmpty()) {
|
if (result.isEmpty()) {
|
||||||
result.reserve(524);
|
result.reserve(524);
|
||||||
|
@ -16743,10 +16743,10 @@ EmojiPack GetPack(DBIEmojiTab tab) {
|
||||||
return result;
|
return result;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case dbietRecent: {
|
case dbiesRecent: {
|
||||||
auto result = EmojiPack();
|
auto result = EmojiPack();
|
||||||
result.reserve(cGetRecentEmoji().size());
|
result.reserve(GetRecent().size());
|
||||||
for (auto &item : cGetRecentEmoji()) {
|
for (auto &item : GetRecent()) {
|
||||||
result.push_back(item.first);
|
result.push_back(item.first);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -51,6 +51,8 @@ EmojiPtr FindReplace(const QChar *ch, const QChar *end, int *outLength = nullptr
|
||||||
void Init();
|
void Init();
|
||||||
|
|
||||||
constexpr auto kPostfix = static_cast<ushort>(0xFE0F);
|
constexpr auto kPostfix = static_cast<ushort>(0xFE0F);
|
||||||
|
constexpr auto kPanPerRow = 7;
|
||||||
|
constexpr auto kPanRowsPerPage = 6;
|
||||||
|
|
||||||
class One {
|
class One {
|
||||||
struct CreationTag {
|
struct CreationTag {
|
||||||
|
@ -219,8 +221,8 @@ inline QString Filename(int index = Index()) {
|
||||||
return QString::fromLatin1(EmojiNames[index]);
|
return QString::fromLatin1(EmojiNames[index]);
|
||||||
}
|
}
|
||||||
|
|
||||||
int GetPackCount(DBIEmojiTab tab);
|
int GetPackCount(DBIEmojiSection tab);
|
||||||
EmojiPack GetPack(DBIEmojiTab tab);
|
EmojiPack GetPack(DBIEmojiSection tab);
|
||||||
|
|
||||||
inline void appendPartToResult(QString &result, const QChar *start, const QChar *from, const QChar *to, EntitiesInText *inOutEntities) {
|
inline void appendPartToResult(QString &result, const QChar *start, const QChar *from, const QChar *to, EntitiesInText *inOutEntities) {
|
||||||
if (to > from) {
|
if (to > from) {
|
||||||
|
@ -289,5 +291,78 @@ inline QString ReplaceInText(const QString &text, EntitiesInText *inOutEntities)
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline RecentEmojiPack &GetRecent() {
|
||||||
|
if (cRecentEmoji().isEmpty()) {
|
||||||
|
RecentEmojiPack result;
|
||||||
|
auto haveAlready = [&result](EmojiPtr emoji) {
|
||||||
|
for (auto &row : result) {
|
||||||
|
if (row.first->id() == emoji->id()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
if (!cRecentEmojiPreload().isEmpty()) {
|
||||||
|
auto preload = cRecentEmojiPreload();
|
||||||
|
cSetRecentEmojiPreload(RecentEmojiPreload());
|
||||||
|
result.reserve(preload.size());
|
||||||
|
for (auto i = preload.cbegin(), e = preload.cend(); i != e; ++i) {
|
||||||
|
if (auto emoji = Ui::Emoji::Find(i->first)) {
|
||||||
|
if (!haveAlready(emoji)) {
|
||||||
|
result.push_back(qMakePair(emoji, i->second));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
uint64 defaultRecent[] = {
|
||||||
|
0xD83DDE02LLU,
|
||||||
|
0xD83DDE18LLU,
|
||||||
|
0x2764LLU,
|
||||||
|
0xD83DDE0DLLU,
|
||||||
|
0xD83DDE0ALLU,
|
||||||
|
0xD83DDE01LLU,
|
||||||
|
0xD83DDC4DLLU,
|
||||||
|
0x263ALLU,
|
||||||
|
0xD83DDE14LLU,
|
||||||
|
0xD83DDE04LLU,
|
||||||
|
0xD83DDE2DLLU,
|
||||||
|
0xD83DDC8BLLU,
|
||||||
|
0xD83DDE12LLU,
|
||||||
|
0xD83DDE33LLU,
|
||||||
|
0xD83DDE1CLLU,
|
||||||
|
0xD83DDE48LLU,
|
||||||
|
0xD83DDE09LLU,
|
||||||
|
0xD83DDE03LLU,
|
||||||
|
0xD83DDE22LLU,
|
||||||
|
0xD83DDE1DLLU,
|
||||||
|
0xD83DDE31LLU,
|
||||||
|
0xD83DDE21LLU,
|
||||||
|
0xD83DDE0FLLU,
|
||||||
|
0xD83DDE1ELLU,
|
||||||
|
0xD83DDE05LLU,
|
||||||
|
0xD83DDE1ALLU,
|
||||||
|
0xD83DDE4ALLU,
|
||||||
|
0xD83DDE0CLLU,
|
||||||
|
0xD83DDE00LLU,
|
||||||
|
0xD83DDE0BLLU,
|
||||||
|
0xD83DDE06LLU,
|
||||||
|
0xD83DDC4CLLU,
|
||||||
|
0xD83DDE10LLU,
|
||||||
|
0xD83DDE15LLU,
|
||||||
|
};
|
||||||
|
for (auto oldKey : defaultRecent) {
|
||||||
|
if (result.size() >= kPanPerRow * kPanRowsPerPage) break;
|
||||||
|
|
||||||
|
if (auto emoji = Ui::Emoji::FromOldKey(oldKey)) {
|
||||||
|
if (!haveAlready(emoji)) {
|
||||||
|
result.push_back(qMakePair(emoji, 1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cSetRecentEmoji(result);
|
||||||
|
}
|
||||||
|
return cRefRecentEmoji();
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace Emoji
|
} // namespace Emoji
|
||||||
} // namespace Ui
|
} // namespace Ui
|
||||||
|
|
|
@ -236,11 +236,12 @@ QImage SettingsSlider::prepareRippleMask(int sectionIndex, const Section §io
|
||||||
void SettingsSlider::paintEvent(QPaintEvent *e) {
|
void SettingsSlider::paintEvent(QPaintEvent *e) {
|
||||||
Painter p(this);
|
Painter p(this);
|
||||||
|
|
||||||
|
auto clip = e->rect();
|
||||||
auto ms = getms();
|
auto ms = getms();
|
||||||
auto activeLeft = getCurrentActiveLeft(ms);
|
auto activeLeft = getCurrentActiveLeft(ms);
|
||||||
|
|
||||||
p.setFont(_st.labelFont);
|
p.setFont(_st.labelFont);
|
||||||
enumerateSections([this, &p, activeLeft, ms](Section §ion) {
|
enumerateSections([this, &p, activeLeft, ms, clip](Section §ion) {
|
||||||
auto active = 1. - snap(qAbs(activeLeft - section.left) / float64(section.width), 0., 1.);
|
auto active = 1. - snap(qAbs(activeLeft - section.left) / float64(section.width), 0., 1.);
|
||||||
if (section.ripple) {
|
if (section.ripple) {
|
||||||
auto color = anim::color(_st.rippleBg, _st.rippleBgActive, active);
|
auto color = anim::color(_st.rippleBg, _st.rippleBgActive, active);
|
||||||
|
@ -266,8 +267,10 @@ void SettingsSlider::paintEvent(QPaintEvent *e) {
|
||||||
if (tofill) {
|
if (tofill) {
|
||||||
p.fillRect(myrtlrect(from, _st.barTop, tofill, _st.barStroke), _st.barFg);
|
p.fillRect(myrtlrect(from, _st.barTop, tofill, _st.barStroke), _st.barFg);
|
||||||
}
|
}
|
||||||
p.setPen(anim::pen(_st.labelFg, _st.labelFgActive, active));
|
if (myrtlrect(section.left, _st.labelTop, section.width, _st.labelFont->height).intersects(clip)) {
|
||||||
p.drawTextLeft(section.left + (section.width - section.labelWidth) / 2, _st.labelTop, width(), section.label, section.labelWidth);
|
p.setPen(anim::pen(_st.labelFg, _st.labelFgActive, active));
|
||||||
|
p.drawTextLeft(section.left + (section.width - section.labelWidth) / 2, _st.labelTop, width(), section.label, section.labelWidth);
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,10 @@ if "%Command%" == "header" (
|
||||||
) else if "%Command%" == "source" (
|
) else if "%Command%" == "source" (
|
||||||
call :write_source %2
|
call :write_source %2
|
||||||
exit /b %errorlevel%
|
exit /b %errorlevel%
|
||||||
|
) else if "%Command%" == "" (
|
||||||
|
echo This is an utility for fast blank module creation.
|
||||||
|
echo Please provide module path.
|
||||||
|
exit /b
|
||||||
)
|
)
|
||||||
|
|
||||||
call :write_module %Command%
|
call :write_module %Command%
|
||||||
|
|
|
@ -74,6 +74,7 @@
|
||||||
],
|
],
|
||||||
|
|
||||||
'dependencies': [
|
'dependencies': [
|
||||||
|
'codegen.gyp:codegen_emoji',
|
||||||
'codegen.gyp:codegen_style',
|
'codegen.gyp:codegen_style',
|
||||||
'codegen.gyp:codegen_numbers',
|
'codegen.gyp:codegen_numbers',
|
||||||
'codegen.gyp:MetaLang',
|
'codegen.gyp:MetaLang',
|
||||||
|
|
Loading…
Reference in New Issue