Move EmptyUserpic from data_peer to empty_userpic.

This commit is contained in:
John Preston 2017-12-05 12:43:18 +04:00
parent 68009b6fba
commit 5eeb8143b6
14 changed files with 346 additions and 303 deletions

View File

@ -30,6 +30,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "ui/widgets/buttons.h" #include "ui/widgets/buttons.h"
#include "ui/widgets/labels.h" #include "ui/widgets/labels.h"
#include "ui/toast/toast.h" #include "ui/toast/toast.h"
#include "ui/empty_userpic.h"
#include "core/click_handler_types.h" #include "core/click_handler_types.h"
#include "storage/localstorage.h" #include "storage/localstorage.h"
#include "auth_session.h" #include "auth_session.h"
@ -643,7 +644,9 @@ ConfirmInviteBox::ConfirmInviteBox(QWidget*, const QString &title, bool isChanne
} }
} }
if (!_photo) { if (!_photo) {
_photoEmpty.set(0, title); _photoEmpty = std::make_unique<Ui::EmptyUserpic>(
Data::PeerUserpicColor(0),
title);
} }
} }
@ -693,7 +696,7 @@ void ConfirmInviteBox::paintEvent(QPaintEvent *e) {
if (_photo) { if (_photo) {
p.drawPixmap((width() - st::confirmInvitePhotoSize) / 2, st::confirmInvitePhotoTop, _photo->pixCircled(st::confirmInvitePhotoSize, st::confirmInvitePhotoSize)); p.drawPixmap((width() - st::confirmInvitePhotoSize) / 2, st::confirmInvitePhotoTop, _photo->pixCircled(st::confirmInvitePhotoSize, st::confirmInvitePhotoSize));
} else { } else {
_photoEmpty.paint(p, (width() - st::confirmInvitePhotoSize) / 2, st::confirmInvitePhotoTop, width(), st::confirmInvitePhotoSize); _photoEmpty->paint(p, (width() - st::confirmInvitePhotoSize) / 2, st::confirmInvitePhotoTop, width(), st::confirmInvitePhotoSize);
} }
int sumWidth = _participants.size() * _userWidth; int sumWidth = _participants.size() * _userWidth;
@ -703,3 +706,5 @@ void ConfirmInviteBox::paintEvent(QPaintEvent *e) {
left += _userWidth; left += _userWidth;
} }
} }
ConfirmInviteBox::~ConfirmInviteBox() = default;

View File

@ -25,6 +25,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
namespace Ui { namespace Ui {
class Checkbox; class Checkbox;
class FlatLabel; class FlatLabel;
class EmptyUserpic;
} // namespace Ui } // namespace Ui
class InformBox; class InformBox;
@ -207,6 +208,7 @@ private:
class ConfirmInviteBox : public BoxContent, public RPCSender { class ConfirmInviteBox : public BoxContent, public RPCSender {
public: public:
ConfirmInviteBox(QWidget*, const QString &title, bool isChannel, const MTPChatPhoto &photo, int count, const QVector<UserData*> &participants); ConfirmInviteBox(QWidget*, const QString &title, bool isChannel, const MTPChatPhoto &photo, int count, const QVector<UserData*> &participants);
~ConfirmInviteBox();
protected: protected:
void prepare() override; void prepare() override;
@ -218,7 +220,7 @@ private:
object_ptr<Ui::FlatLabel> _title; object_ptr<Ui::FlatLabel> _title;
object_ptr<Ui::FlatLabel> _status; object_ptr<Ui::FlatLabel> _status;
ImagePtr _photo; ImagePtr _photo;
EmptyUserpic _photoEmpty; std::unique_ptr<Ui::EmptyUserpic> _photoEmpty;
QVector<UserData*> _participants; QVector<UserData*> _participants;
int _userWidth = 0; int _userWidth = 0;

View File

@ -28,6 +28,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "ui/widgets/checkbox.h" #include "ui/widgets/checkbox.h"
#include "ui/widgets/buttons.h" #include "ui/widgets/buttons.h"
#include "ui/widgets/input_fields.h" #include "ui/widgets/input_fields.h"
#include "ui/empty_userpic.h"
#include "styles/style_history.h" #include "styles/style_history.h"
#include "styles/style_boxes.h" #include "styles/style_boxes.h"
#include "media/media_clip_reader.h" #include "media/media_clip_reader.h"
@ -235,7 +236,9 @@ SendFilesBox::SendFilesBox(QWidget*, const QString &phone, const QString &firstn
_nameText.setText(st::semiboldTextStyle, name, _textNameOptions); _nameText.setText(st::semiboldTextStyle, name, _textNameOptions);
_statusText = _contactPhone; _statusText = _contactPhone;
_statusWidth = qMax(_nameText.maxWidth(), st::normalFont->width(_statusText)); _statusWidth = qMax(_nameText.maxWidth(), st::normalFont->width(_statusText));
_contactPhotoEmpty.set(0, name); _contactPhotoEmpty = std::make_unique<Ui::EmptyUserpic>(
Data::PeerUserpicColor(0),
name);
} }
void SendFilesBox::prepare() { void SendFilesBox::prepare() {
@ -400,7 +403,7 @@ void SendFilesBox::paintEvent(QPaintEvent *e) {
auto &icon = _fileIsAudio ? st::historyFileOutPlay : _fileIsImage ? st::historyFileOutImage : st::historyFileOutDocument; auto &icon = _fileIsAudio ? st::historyFileOutPlay : _fileIsImage ? st::historyFileOutImage : st::historyFileOutDocument;
icon.paintInCenter(p, inner); icon.paintInCenter(p, inner);
} else { } else {
_contactPhotoEmpty.paint(p, x + st::msgFilePadding.left(), y + st::msgFilePadding.top(), width(), st::msgFileSize); _contactPhotoEmpty->paint(p, x + st::msgFilePadding.left(), y + st::msgFilePadding.top(), width(), st::msgFileSize);
} }
} else { } else {
QRect rthumb(rtlrect(x + st::msgFileThumbPadding.left(), y + st::msgFileThumbPadding.top(), st::msgFileThumbSize, st::msgFileThumbSize, width())); QRect rthumb(rtlrect(x + st::msgFileThumbPadding.left(), y + st::msgFileThumbPadding.top(), st::msgFileThumbSize, st::msgFileThumbSize, width()));
@ -457,6 +460,8 @@ void SendFilesBox::onSend(bool ctrlShiftEnter) {
closeBox(); closeBox();
} }
SendFilesBox::~SendFilesBox() = default;
EditCaptionBox::EditCaptionBox(QWidget*, HistoryMedia *media, FullMsgId msgId) : _msgId(msgId) { EditCaptionBox::EditCaptionBox(QWidget*, HistoryMedia *media, FullMsgId msgId) : _msgId(msgId) {
Expects(media->canEditCaption()); Expects(media->canEditCaption());

View File

@ -27,6 +27,7 @@ namespace Ui {
class Checkbox; class Checkbox;
class RoundButton; class RoundButton;
class InputArea; class InputArea;
class EmptyUserpic;
} // namespace Ui } // namespace Ui
class SendFilesBox : public BoxContent { class SendFilesBox : public BoxContent {
@ -44,6 +45,8 @@ public:
_cancelledCallback = std::move(callback); _cancelledCallback = std::move(callback);
} }
~SendFilesBox();
protected: protected:
void prepare() override; void prepare() override;
void setInnerFocus() override; void setInnerFocus() override;
@ -96,7 +99,7 @@ private:
QString _contactPhone; QString _contactPhone;
QString _contactFirstName; QString _contactFirstName;
QString _contactLastName; QString _contactLastName;
EmptyUserpic _contactPhotoEmpty; std::unique_ptr<Ui::EmptyUserpic> _contactPhotoEmpty;
base::lambda<void(const QStringList &files, const QImage &image, std::unique_ptr<FileLoadTask::MediaInformation> information, bool compressed, const QString &caption, bool ctrlShiftEnter)> _confirmedCallback; base::lambda<void(const QStringList &files, const QImage &image, std::unique_ptr<FileLoadTask::MediaInformation> information, bool compressed, const QString &caption, bool ctrlShiftEnter)> _confirmedCallback;
base::lambda<void()> _cancelledCallback; base::lambda<void()> _cancelledCallback;

View File

@ -29,6 +29,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "ui/widgets/shadow.h" #include "ui/widgets/shadow.h"
#include "ui/effects/ripple_animation.h" #include "ui/effects/ripple_animation.h"
#include "ui/wrap/fade_wrap.h" #include "ui/wrap/fade_wrap.h"
#include "ui/empty_userpic.h"
#include "messenger.h" #include "messenger.h"
#include "mainwindow.h" #include "mainwindow.h"
#include "lang/lang_keys.h" #include "lang/lang_keys.h"
@ -484,7 +485,10 @@ void Panel::createUserpicCache(ImagePtr image) {
filled.setDevicePixelRatio(cRetinaFactor()); filled.setDevicePixelRatio(cRetinaFactor());
{ {
Painter p(&filled); Painter p(&filled);
EmptyUserpic(_user->id, _user->name).paintSquare(p, 0, 0, st::callWidth, st::callWidth); Ui::EmptyUserpic(
Data::PeerUserpicColor(_user->id),
_user->name
).paintSquare(p, 0, 0, st::callWidth, st::callWidth);
} }
Images::prepareRound(filled, ImageRoundRadius::Large, ImageRoundCorner::TopLeft | ImageRoundCorner::TopRight); Images::prepareRound(filled, ImageRoundRadius::Large, ImageRoundCorner::TopLeft | ImageRoundCorner::TopRight);
_userPhoto = App::pixmapFromImageInPlace(std::move(filled)); _userPhoto = App::pixmapFromImageInPlace(std::move(filled));

View File

@ -35,29 +35,25 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "messenger.h" #include "messenger.h"
#include "mainwindow.h" #include "mainwindow.h"
#include "window/window_controller.h" #include "window/window_controller.h"
#include "ui/empty_userpic.h"
namespace { namespace {
constexpr auto kUpdateFullPeerTimeout = TimeMs(5000); // Not more than once in 5 seconds. constexpr auto kUpdateFullPeerTimeout = TimeMs(5000); // Not more than once in 5 seconds.
constexpr auto kUserpicSize = 160; constexpr auto kUserpicSize = 160;
int peerColorIndex(const PeerId &peer) { } // namespace
auto myId = Auth().userId();
auto peerId = peerToBareInt(peer); namespace Data {
auto both = (QByteArray::number(peerId) + QByteArray::number(myId)).mid(0, 15);
uchar md5[16]; int PeerColorIndex(int32 bareId) {
hashMd5(both.constData(), both.size(), md5); const auto index = std::abs(bareId) % 7;
return (md5[peerId & 0x0F] & (peerIsUser(peer) ? 0x07 : 0x03)); const int map[] = { 0, 7, 4, 1, 6, 3, 5 };
return map[index];
} }
ImagePtr generateUserpicImage(const style::icon &icon) { int PeerColorIndex(PeerId peerId) {
auto data = QImage(icon.size() * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied); return PeerColorIndex(peerToBareInt(peerId));
data.setDevicePixelRatio(cRetinaFactor());
{
Painter p(&data);
icon.paint(p, 0, 0, icon.width());
}
return ImagePtr(App::pixmapFromImageInPlace(std::move(data)), "PNG");
} }
style::color PeerUserpicColor(PeerId peerId) { style::color PeerUserpicColor(PeerId peerId) {
@ -74,190 +70,7 @@ style::color PeerUserpicColor(PeerId peerId) {
return colors[PeerColorIndex(peerId)]; return colors[PeerColorIndex(peerId)];
} }
} // namespace } // namespace Data
int PeerColorIndex(int32 bareId) {
const auto index = std::abs(bareId) % 7;
const int map[] = { 0, 7, 4, 1, 6, 3, 5 };
return map[index];
}
int PeerColorIndex(PeerId peerId) {
return PeerColorIndex(peerToBareInt(peerId));
}
class EmptyUserpic::Impl {
public:
Impl(PeerId peerId, const QString &name)
: _color(PeerUserpicColor(peerId)) {
fillString(name);
}
void paint(Painter &p, int x, int y, int size);
void paintRounded(Painter &p, int x, int y, int size);
void paintSquare(Painter &p, int x, int y, int size);
StorageKey uniqueKey() const;
private:
template <typename PaintBackground>
void paint(Painter &p, int x, int y, int size, PaintBackground paintBackground);
void fillString(const QString &name);
style::color _color;
QString _string;
};
template <typename PaintBackground>
void EmptyUserpic::Impl::paint(Painter &p, int x, int y, int size, PaintBackground paintBackground) {
auto fontsize = (size * 13) / 33;
auto font = st::historyPeerUserpicFont->f;
font.setPixelSize(fontsize);
PainterHighQualityEnabler hq(p);
p.setBrush(_color);
p.setPen(Qt::NoPen);
paintBackground();
p.setFont(font);
p.setBrush(Qt::NoBrush);
p.setPen(st::historyPeerUserpicFg);
p.drawText(QRect(x, y, size, size), _string, QTextOption(style::al_center));
}
void EmptyUserpic::Impl::paint(Painter &p, int x, int y, int size) {
paint(p, x, y, size, [&p, x, y, size] {
p.drawEllipse(x, y, size, size);
});
}
void EmptyUserpic::Impl::paintRounded(Painter &p, int x, int y, int size) {
paint(p, x, y, size, [&p, x, y, size] {
p.drawRoundedRect(x, y, size, size, st::buttonRadius, st::buttonRadius);
});
}
void EmptyUserpic::Impl::paintSquare(Painter &p, int x, int y, int size) {
paint(p, x, y, size, [&p, x, y, size] {
p.fillRect(x, y, size, size, p.brush());
});
}
StorageKey EmptyUserpic::Impl::uniqueKey() const {
auto first = 0xFFFFFFFF00000000ULL | anim::getPremultiplied(_color->c);
auto second = uint64(0);
memcpy(&second, _string.constData(), qMin(sizeof(second), _string.size() * sizeof(QChar)));
return StorageKey(first, second);
}
void EmptyUserpic::Impl::fillString(const QString &name) {
QList<QString> letters;
QList<int> levels;
auto level = 0;
auto letterFound = false;
auto ch = name.constData(), end = ch + name.size();
while (ch != end) {
auto emojiLength = 0;
if (auto emoji = Ui::Emoji::Find(ch, end, &emojiLength)) {
ch += emojiLength;
} else if (ch->isHighSurrogate()) {
++ch;
if (ch != end && ch->isLowSurrogate()) {
++ch;
}
} else if (!letterFound && ch->isLetterOrNumber()) {
letterFound = true;
if (ch + 1 != end && chIsDiac(*(ch + 1))) {
letters.push_back(QString(ch, 2));
levels.push_back(level);
++ch;
} else {
letters.push_back(QString(ch, 1));
levels.push_back(level);
}
++ch;
} else {
if (*ch == ' ') {
level = 0;
letterFound = false;
} else if (letterFound && *ch == '-') {
level = 1;
letterFound = true;
}
++ch;
}
}
// We prefer the second letter to be after ' ', but it can also be after '-'.
_string = QString();
if (!letters.isEmpty()) {
_string += letters.front();
auto bestIndex = 0;
auto bestLevel = 2;
for (auto i = letters.size(); i != 1;) {
if (levels[--i] < bestLevel) {
bestIndex = i;
bestLevel = levels[i];
}
}
if (bestIndex > 0) {
_string += letters[bestIndex];
}
}
_string = _string.toUpper();
}
EmptyUserpic::EmptyUserpic() = default;
EmptyUserpic::EmptyUserpic(PeerId peerId, const QString &name)
: _impl(std::make_unique<Impl>(peerId, name)) {
}
EmptyUserpic::EmptyUserpic(const QString &nonce, const QString &name)
: EmptyUserpic(qHash(nonce), name) {
}
void EmptyUserpic::set(PeerId peerId, const QString &name) {
_impl = std::make_unique<Impl>(peerId, name);
}
void EmptyUserpic::clear() {
_impl.reset();
}
void EmptyUserpic::paint(Painter &p, int x, int y, int outerWidth, int size) const {
Expects(_impl != nullptr);
_impl->paint(p, rtl() ? (outerWidth - x - size) : x, y, size);
}
void EmptyUserpic::paintRounded(Painter &p, int x, int y, int outerWidth, int size) const {
Expects(_impl != nullptr);
_impl->paintRounded(p, rtl() ? (outerWidth - x - size) : x, y, size);
}
void EmptyUserpic::paintSquare(Painter &p, int x, int y, int outerWidth, int size) const {
Expects(_impl != nullptr);
_impl->paintSquare(p, rtl() ? (outerWidth - x - size) : x, y, size);
}
StorageKey EmptyUserpic::uniqueKey() const {
Expects(_impl != nullptr);
return _impl->uniqueKey();
}
QPixmap EmptyUserpic::generate(int size) {
auto result = QImage(QSize(size, size) * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied);
result.setDevicePixelRatio(cRetinaFactor());
result.fill(Qt::transparent);
{
Painter p(&result);
paint(p, 0, 0, size, size);
}
return App::pixmapFromImageInPlace(std::move(result));
}
EmptyUserpic::~EmptyUserpic() = default;
using UpdateFlag = Notify::PeerUpdate::Flag; using UpdateFlag = Notify::PeerUpdate::Flag;
@ -285,13 +98,11 @@ void PeerClickHandler::onClick(Qt::MouseButton button) const {
} }
PeerData::PeerData(const PeerId &id) PeerData::PeerData(const PeerId &id)
: id(id) { : id(id)
, _userpicEmpty(createEmptyUserpic()) {
nameText.setText(st::msgNameStyle, QString(), _textNameOptions); nameText.setText(st::msgNameStyle, QString(), _textNameOptions);
_userpicEmpty.set(id, QString());
} }
PeerData::~PeerData() = default;
void PeerData::updateNameDelayed( void PeerData::updateNameDelayed(
const QString &newName, const QString &newName,
const QString &newNameOrPhone, const QString &newNameOrPhone,
@ -314,9 +125,7 @@ void PeerData::updateNameDelayed(
++nameVersion; ++nameVersion;
name = newName; name = newName;
nameText.setText(st::msgNameStyle, name, _textNameOptions); nameText.setText(st::msgNameStyle, name, _textNameOptions);
if (useEmptyUserpic()) { refreshEmptyUserpic();
_userpicEmpty.set(id, name);
}
Notify::PeerUpdate update(this); Notify::PeerUpdate update(this);
update.flags |= UpdateFlag::NameChanged; update.flags |= UpdateFlag::NameChanged;
@ -344,6 +153,16 @@ void PeerData::updateNameDelayed(
Notify::PeerUpdated().notify(update, true); Notify::PeerUpdated().notify(update, true);
} }
std::unique_ptr<Ui::EmptyUserpic> PeerData::createEmptyUserpic() const {
return std::make_unique<Ui::EmptyUserpic>(
Data::PeerUserpicColor(id),
name);
}
void PeerData::refreshEmptyUserpic() const {
_userpicEmpty = useEmptyUserpic() ? createEmptyUserpic() : nullptr;
}
ClickHandlerPtr PeerData::createOpenLink() { ClickHandlerPtr PeerData::createOpenLink() {
return MakeShared<PeerClickHandler>(this); return MakeShared<PeerClickHandler>(this);
} }
@ -355,11 +174,7 @@ void PeerData::setUserpic(
_userpicPhotoId = photoId; _userpicPhotoId = photoId;
_userpic = userpic; _userpic = userpic;
_userpicLocation = location; _userpicLocation = location;
if (useEmptyUserpic()) { refreshEmptyUserpic();
_userpicEmpty.set(id, name);
} else {
_userpicEmpty.clear();
}
} }
void PeerData::setUserpicPhoto(const MTPPhoto &data) { void PeerData::setUserpicPhoto(const MTPPhoto &data) {
@ -381,7 +196,7 @@ ImagePtr PeerData::currentUserpic() const {
_userpic->load(); _userpic->load();
if (_userpic->loaded()) { if (_userpic->loaded()) {
if (!useEmptyUserpic()) { if (!useEmptyUserpic()) {
_userpicEmpty.clear(); _userpicEmpty = nullptr;
} }
return _userpic; return _userpic;
} }
@ -393,7 +208,7 @@ void PeerData::paintUserpic(Painter &p, int x, int y, int size) const {
if (auto userpic = currentUserpic()) { if (auto userpic = currentUserpic()) {
p.drawPixmap(x, y, userpic->pixCircled(size, size)); p.drawPixmap(x, y, userpic->pixCircled(size, size));
} else { } else {
_userpicEmpty.paint(p, x, y, x + size + x, size); _userpicEmpty->paint(p, x, y, x + size + x, size);
} }
} }
@ -401,7 +216,7 @@ void PeerData::paintUserpicRounded(Painter &p, int x, int y, int size) const {
if (auto userpic = currentUserpic()) { if (auto userpic = currentUserpic()) {
p.drawPixmap(x, y, userpic->pixRounded(size, size, ImageRoundRadius::Small)); p.drawPixmap(x, y, userpic->pixRounded(size, size, ImageRoundRadius::Small));
} else { } else {
_userpicEmpty.paintRounded(p, x, y, x + size + x, size); _userpicEmpty->paintRounded(p, x, y, x + size + x, size);
} }
} }
@ -409,13 +224,13 @@ void PeerData::paintUserpicSquare(Painter &p, int x, int y, int size) const {
if (auto userpic = currentUserpic()) { if (auto userpic = currentUserpic()) {
p.drawPixmap(x, y, userpic->pix(size, size)); p.drawPixmap(x, y, userpic->pix(size, size));
} else { } else {
_userpicEmpty.paintSquare(p, x, y, x + size + x, size); _userpicEmpty->paintSquare(p, x, y, x + size + x, size);
} }
} }
StorageKey PeerData::userpicUniqueKey() const { StorageKey PeerData::userpicUniqueKey() const {
if (useEmptyUserpic()) { if (useEmptyUserpic()) {
return _userpicEmpty.uniqueKey(); return _userpicEmpty->uniqueKey();
} }
return storageKey(_userpicLocation); return storageKey(_userpicLocation);
} }
@ -456,17 +271,6 @@ QPixmap PeerData::genUserpicRounded(int size) const {
return App::pixmapFromImageInPlace(std::move(result)); return App::pixmapFromImageInPlace(std::move(result));
} }
const Text &BotCommand::descriptionText() const {
if (_descriptionText.isEmpty() && !_description.isEmpty()) {
_descriptionText.setText(st::defaultTextStyle, _description, _textNameOptions);
}
return _descriptionText;
}
bool UserData::canShareThisContact() const {
return canShareThisContactFast() || !App::phoneFromSharedContact(peerToUser(id)).isEmpty();
}
void PeerData::updateUserpic( void PeerData::updateUserpic(
PhotoId photoId, PhotoId photoId,
const MTPFileLocation &location) { const MTPFileLocation &location) {
@ -498,16 +302,6 @@ void PeerData::clearUserpic() {
}(); }();
} }
// see Local::readPeer as well
void UserData::setPhoto(const MTPUserProfilePhoto &photo) {
if (photo.type() == mtpc_userProfilePhoto) {
const auto &data = photo.c_userProfilePhoto();
updateUserpic(data.vphoto_id.v, data.vphoto_small);
} else {
clearUserpic();
}
}
void PeerData::fillNames() { void PeerData::fillNames() {
_nameWords.clear(); _nameWords.clear();
_nameFirstChars.clear(); _nameFirstChars.clear();
@ -530,6 +324,29 @@ void PeerData::fillNames() {
} }
} }
PeerData::~PeerData() = default;
const Text &BotCommand::descriptionText() const {
if (_descriptionText.isEmpty() && !_description.isEmpty()) {
_descriptionText.setText(st::defaultTextStyle, _description, _textNameOptions);
}
return _descriptionText;
}
bool UserData::canShareThisContact() const {
return canShareThisContactFast() || !App::phoneFromSharedContact(peerToUser(id)).isEmpty();
}
// see Local::readPeer as well
void UserData::setPhoto(const MTPUserProfilePhoto &photo) {
if (photo.type() == mtpc_userProfilePhoto) {
const auto &data = photo.c_userProfilePhoto();
updateUserpic(data.vphoto_id.v, data.vphoto_small);
} else {
clearUserpic();
}
}
bool UserData::setAbout(const QString &newAbout) { bool UserData::setAbout(const QString &newAbout) {
if (_about == newAbout) { if (_about == newAbout) {
return false; return false;

View File

@ -24,53 +24,22 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "data/data_flags.h" #include "data/data_flags.h"
#include "data/data_notify_settings.h" #include "data/data_notify_settings.h"
int PeerColorIndex(PeerId peerId); namespace Ui {
int PeerColorIndex(int32 bareId); class EmptyUserpic;
} // namespace Ui
class EmptyUserpic {
public:
EmptyUserpic();
EmptyUserpic(PeerId peerId, const QString &name);
EmptyUserpic(const QString &nonce, const QString &name);
void set(PeerId peerId, const QString &name);
void clear();
explicit operator bool() const {
return (_impl != nullptr);
}
void paint(
Painter &p,
int x,
int y,
int outerWidth,
int size) const;
void paintRounded(
Painter &p,
int x,
int y,
int outerWidth,
int size) const;
void paintSquare(
Painter &p,
int x,
int y,
int outerWidth,
int size) const;
QPixmap generate(int size);
StorageKey uniqueKey() const;
~EmptyUserpic();
private:
class Impl;
std::unique_ptr<Impl> _impl;
friend class Impl;
};
class PeerData; class PeerData;
class UserData;
class ChatData;
class ChannelData;
namespace Data {
int PeerColorIndex(PeerId peerId);
int PeerColorIndex(int32 bareId);
style::color PeerUserpicColor(PeerId peerId);
} // namespace Data
class PeerClickHandler : public ClickHandler { class PeerClickHandler : public ClickHandler {
public: public:
@ -86,10 +55,6 @@ private:
}; };
class UserData;
class ChatData;
class ChannelData;
class PeerData { class PeerData {
protected: protected:
PeerData(const PeerId &id); PeerData(const PeerId &id);
@ -270,12 +235,14 @@ protected:
private: private:
void fillNames(); void fillNames();
std::unique_ptr<Ui::EmptyUserpic> createEmptyUserpic() const;
void refreshEmptyUserpic() const;
static constexpr auto kUnknownPhotoId = PhotoId(0xFFFFFFFFFFFFFFFFULL); static constexpr auto kUnknownPhotoId = PhotoId(0xFFFFFFFFFFFFFFFFULL);
ImagePtr _userpic; ImagePtr _userpic;
PhotoId _userpicPhotoId = kUnknownPhotoId; PhotoId _userpicPhotoId = kUnknownPhotoId;
mutable EmptyUserpic _userpicEmpty; mutable std::unique_ptr<Ui::EmptyUserpic> _userpicEmpty;
StorageImageLocation _userpicLocation; StorageImageLocation _userpicLocation;
Data::NotifySettings _notify; Data::NotifySettings _notify;

View File

@ -38,6 +38,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "window/window_controller.h" #include "window/window_controller.h"
#include "styles/style_history.h" #include "styles/style_history.h"
#include "calls/calls_instance.h" #include "calls/calls_instance.h"
#include "ui/empty_userpic.h"
namespace { namespace {
@ -2981,8 +2982,8 @@ void HistoryContact::initDimensions() {
if (_contact) { if (_contact) {
_contact->loadUserpic(); _contact->loadUserpic();
} else { } else {
_photoEmpty.set( _photoEmpty = std::make_unique<Ui::EmptyUserpic>(
_userId ? _userId : _parent->id, Data::PeerUserpicColor(_userId ? _userId : _parent->id),
_name.originalText()); _name.originalText());
} }
if (_contact && _contact->contact > 0) { if (_contact && _contact->contact > 0) {
@ -3046,7 +3047,7 @@ void HistoryContact::draw(Painter &p, const QRect &r, TextSelection selection, T
if (_contact) { if (_contact) {
_contact->paintUserpic(p, rthumb.x(), rthumb.y(), st::msgFileThumbSize); _contact->paintUserpic(p, rthumb.x(), rthumb.y(), st::msgFileThumbSize);
} else { } else {
_photoEmpty.paint(p, st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, width, st::msgFileThumbSize); _photoEmpty->paint(p, st::msgFileThumbPadding.left(), st::msgFileThumbPadding.top() - topMinus, width, st::msgFileThumbSize);
} }
if (selected) { if (selected) {
PainterHighQualityEnabler hq(p); PainterHighQualityEnabler hq(p);
@ -3065,7 +3066,7 @@ void HistoryContact::draw(Painter &p, const QRect &r, TextSelection selection, T
nameright = st::msgFilePadding.left(); nameright = st::msgFilePadding.left();
statustop = st::msgFileStatusTop - topMinus; statustop = st::msgFileStatusTop - topMinus;
_photoEmpty.paint(p, st::msgFilePadding.left(), st::msgFilePadding.top() - topMinus, width, st::msgFileSize); _photoEmpty->paint(p, st::msgFilePadding.left(), st::msgFilePadding.top() - topMinus, width, st::msgFileSize);
} }
int32 namewidth = width - nameleft - nameright; int32 namewidth = width - nameleft - nameright;
@ -3133,6 +3134,8 @@ void HistoryContact::updateSentMedia(const MTPMessageMedia &media) {
} }
} }
HistoryContact::~HistoryContact() = default;
HistoryCall::HistoryCall(not_null<HistoryItem*> parent, const MTPDmessageActionPhoneCall &call) : HistoryMedia(parent) HistoryCall::HistoryCall(not_null<HistoryItem*> parent, const MTPDmessageActionPhoneCall &call) : HistoryMedia(parent)
, _reason(GetReason(call)) { , _reason(GetReason(call)) {
if (_parent->out()) { if (_parent->out()) {

View File

@ -33,6 +33,10 @@ class Playback;
} // namespace Clip } // namespace Clip
} // namespace Media } // namespace Media
namespace Ui {
class EmptyUserpic;
} // namespace Ui
void HistoryInitMedia(); void HistoryInitMedia();
class HistoryFileMedia : public HistoryMedia { class HistoryFileMedia : public HistoryMedia {
@ -716,6 +720,8 @@ public:
return _phone; return _phone;
} }
~HistoryContact();
private: private:
int32 _userId = 0; int32 _userId = 0;
UserData *_contact = nullptr; UserData *_contact = nullptr;
@ -723,7 +729,7 @@ private:
int _phonew = 0; int _phonew = 0;
QString _fname, _lname, _phone; QString _fname, _lname, _phone;
Text _name; Text _name;
EmptyUserpic _photoEmpty; std::unique_ptr<Ui::EmptyUserpic> _photoEmpty;
ClickHandlerPtr _linkl; ClickHandlerPtr _linkl;
int _linkw = 0; int _linkw = 0;

View File

@ -66,7 +66,7 @@ style::color FromNameFg(not_null<PeerData*> peer, bool selected) {
st::historyPeer7NameFgSelected, st::historyPeer7NameFgSelected,
st::historyPeer8NameFgSelected, st::historyPeer8NameFgSelected,
}; };
return colors[PeerColorIndex(peer->id)]; return colors[Data::PeerColorIndex(peer->id)];
} else { } else {
const style::color colors[] = { const style::color colors[] = {
st::historyPeer1NameFg, st::historyPeer1NameFg,
@ -78,7 +78,7 @@ style::color FromNameFg(not_null<PeerData*> peer, bool selected) {
st::historyPeer7NameFg, st::historyPeer7NameFg,
st::historyPeer8NameFg, st::historyPeer8NameFg,
}; };
return colors[PeerColorIndex(peer->id)]; return colors[Data::PeerColorIndex(peer->id)];
} }
} }

View File

@ -27,6 +27,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "inline_bots/inline_bot_layout_internal.h" #include "inline_bots/inline_bot_layout_internal.h"
#include "storage/localstorage.h" #include "storage/localstorage.h"
#include "mainwidget.h" #include "mainwidget.h"
#include "ui/empty_userpic.h"
namespace InlineBots { namespace InlineBots {
namespace Layout { namespace Layout {
@ -153,8 +154,8 @@ ImagePtr ItemBase::getResultThumb() const {
QPixmap ItemBase::getResultContactAvatar(int width, int height) const { QPixmap ItemBase::getResultContactAvatar(int width, int height) const {
if (_result->_type == Result::Type::Contact) { if (_result->_type == Result::Type::Contact) {
auto result = EmptyUserpic( auto result = Ui::EmptyUserpic(
_result->_id, Data::PeerUserpicColor(qHash(_result->_id)),
_result->getLayoutTitle() _result->getLayoutTitle()
).generate(width); ).generate(width);
if (result.height() != height * cIntRetinaFactor()) { if (result.height() != height * cIntRetinaFactor()) {

View File

@ -0,0 +1,159 @@
/*
This file is part of Telegram Desktop,
the official desktop version of Telegram messaging app, see https://telegram.org
Telegram Desktop is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
It is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
In addition, as a special exception, the copyright holders give permission
to link the code of portions of this program with the OpenSSL library.
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
*/
#include "ui/empty_userpic.h"
#include "data/data_peer.h"
#include "styles/style_history.h"
namespace Ui {
EmptyUserpic::EmptyUserpic(const style::color &color, const QString &name)
: _color(color) {
fillString(name);
}
template <typename Callback>
void EmptyUserpic::paint(
Painter &p,
int x,
int y,
int outerWidth,
int size,
Callback paintBackground) const {
x = rtl() ? (outerWidth - x - size) : x;
const auto fontsize = (size * 13) / 33;
auto font = st::historyPeerUserpicFont->f;
font.setPixelSize(fontsize);
PainterHighQualityEnabler hq(p);
p.setBrush(_color);
p.setPen(Qt::NoPen);
paintBackground();
p.setFont(font);
p.setBrush(Qt::NoBrush);
p.setPen(st::historyPeerUserpicFg);
p.drawText(QRect(x, y, size, size), _string, QTextOption(style::al_center));
}
void EmptyUserpic::paint(
Painter &p,
int x,
int y,
int outerWidth,
int size) const {
paint(p, x, y, outerWidth, size, [&p, x, y, size] {
p.drawEllipse(x, y, size, size);
});
}
void EmptyUserpic::paintRounded(Painter &p, int x, int y, int outerWidth, int size) const {
paint(p, x, y, outerWidth, size, [&p, x, y, size] {
p.drawRoundedRect(x, y, size, size, st::buttonRadius, st::buttonRadius);
});
}
void EmptyUserpic::paintSquare(Painter &p, int x, int y, int outerWidth, int size) const {
paint(p, x, y, outerWidth, size, [&p, x, y, size] {
p.fillRect(x, y, size, size, p.brush());
});
}
StorageKey EmptyUserpic::uniqueKey() const {
auto first = 0xFFFFFFFF00000000ULL | anim::getPremultiplied(_color->c);
auto second = uint64(0);
memcpy(&second, _string.constData(), qMin(sizeof(second), _string.size() * sizeof(QChar)));
return StorageKey(first, second);
}
QPixmap EmptyUserpic::generate(int size) {
auto result = QImage(QSize(size, size) * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied);
result.setDevicePixelRatio(cRetinaFactor());
result.fill(Qt::transparent);
{
Painter p(&result);
paint(p, 0, 0, size, size);
}
return App::pixmapFromImageInPlace(std::move(result));
}
void EmptyUserpic::fillString(const QString &name) {
QList<QString> letters;
QList<int> levels;
auto level = 0;
auto letterFound = false;
auto ch = name.constData(), end = ch + name.size();
while (ch != end) {
auto emojiLength = 0;
if (auto emoji = Ui::Emoji::Find(ch, end, &emojiLength)) {
ch += emojiLength;
} else if (ch->isHighSurrogate()) {
++ch;
if (ch != end && ch->isLowSurrogate()) {
++ch;
}
} else if (!letterFound && ch->isLetterOrNumber()) {
letterFound = true;
if (ch + 1 != end && chIsDiac(*(ch + 1))) {
letters.push_back(QString(ch, 2));
levels.push_back(level);
++ch;
} else {
letters.push_back(QString(ch, 1));
levels.push_back(level);
}
++ch;
} else {
if (*ch == ' ') {
level = 0;
letterFound = false;
} else if (letterFound && *ch == '-') {
level = 1;
letterFound = true;
}
++ch;
}
}
// We prefer the second letter to be after ' ', but it can also be after '-'.
_string = QString();
if (!letters.isEmpty()) {
_string += letters.front();
auto bestIndex = 0;
auto bestLevel = 2;
for (auto i = letters.size(); i != 1;) {
if (levels[--i] < bestLevel) {
bestIndex = i;
bestLevel = levels[i];
}
}
if (bestIndex > 0) {
_string += letters[bestIndex];
}
}
_string = _string.toUpper();
}
EmptyUserpic::~EmptyUserpic() = default;
} // namespace Ui

View File

@ -0,0 +1,69 @@
/*
This file is part of Telegram Desktop,
the official desktop version of Telegram messaging app, see https://telegram.org
Telegram Desktop is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
It is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
In addition, as a special exception, the copyright holders give permission
to link the code of portions of this program with the OpenSSL library.
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
*/
#pragma once
namespace Ui {
class EmptyUserpic {
public:
EmptyUserpic(const style::color &color, const QString &name);
void paint(
Painter &p,
int x,
int y,
int outerWidth,
int size) const;
void paintRounded(
Painter &p,
int x,
int y,
int outerWidth,
int size) const;
void paintSquare(
Painter &p,
int x,
int y,
int outerWidth,
int size) const;
QPixmap generate(int size);
StorageKey uniqueKey() const;
~EmptyUserpic();
private:
template <typename Callback>
void paint(
Painter &p,
int x,
int y,
int outerWidth,
int size,
Callback paintBackground) const;
void fillString(const QString &name);
style::color _color;
QString _string;
};
} // namespace Ui

View File

@ -594,6 +594,8 @@
<(src_loc)/ui/countryinput.h <(src_loc)/ui/countryinput.h
<(src_loc)/ui/emoji_config.cpp <(src_loc)/ui/emoji_config.cpp
<(src_loc)/ui/emoji_config.h <(src_loc)/ui/emoji_config.h
<(src_loc)/ui/empty_userpic.cpp
<(src_loc)/ui/empty_userpic.h
<(src_loc)/ui/focus_persister.h <(src_loc)/ui/focus_persister.h
<(src_loc)/ui/images.cpp <(src_loc)/ui/images.cpp
<(src_loc)/ui/images.h <(src_loc)/ui/images.h