tdesktop/Telegram/SourceFiles/ui/unread_badge_paint.cpp
2022-12-26 14:24:06 +04:00

184 lines
4.9 KiB
C++

/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "ui/unread_badge_paint.h"
#include "ui/ui_utility.h"
#include "styles/style_dialogs.h"
namespace Ui {
namespace {
struct UnreadBadgeSizeData {
QImage circle;
QPixmap left[6], right[6];
};
class UnreadBadgeStyleData {
public:
UnreadBadgeStyleData();
UnreadBadgeSizeData sizes[static_cast<int>(UnreadBadgeSize::kCount)];
style::color bg[6] = {
st::dialogsUnreadBg,
st::dialogsUnreadBgOver,
st::dialogsUnreadBgActive,
st::dialogsUnreadBgMuted,
st::dialogsUnreadBgMutedOver,
st::dialogsUnreadBgMutedActive
};
style::color reactionBg[6] = {
st::dialogsDraftFg,
st::dialogsDraftFgOver,
st::dialogsDraftFgActive,
st::dialogsUnreadBgMuted,
st::dialogsUnreadBgMutedOver,
st::dialogsUnreadBgMutedActive
};
rpl::lifetime lifetime;
};
UnreadBadgeStyleData::UnreadBadgeStyleData() {
style::PaletteChanged(
) | rpl::start_with_next([=] {
for (auto &data : sizes) {
for (auto &left : data.left) {
left = QPixmap();
}
for (auto &right : data.right) {
right = QPixmap();
}
}
}, lifetime);
}
UnreadBadgeStyleData &UnreadBadgeStyles() {
static auto result = UnreadBadgeStyleData();
return result;
}
void CreateCircleMask(UnreadBadgeSizeData *data, int size) {
if (!data->circle.isNull()) {
return;
}
data->circle = style::createCircleMask(size);
}
[[nodiscard]] QImage ColorizeCircleHalf(
UnreadBadgeSizeData *data,
int size,
int half,
int xoffset,
style::color color) {
auto result = style::colorizeImage(data->circle, color, QRect(xoffset, 0, half, size));
result.setDevicePixelRatio(style::DevicePixelRatio());
return result;
}
[[nodiscard]] QString ComputeUnreadBadgeText(
const QString &unreadCount,
int allowDigits) {
return (allowDigits > 0) && (unreadCount.size() > allowDigits + 1)
? u".."_q + unreadCount.mid(unreadCount.size() - allowDigits)
: unreadCount;
}
void PaintUnreadBadge(QPainter &p, const QRect &rect, const UnreadBadgeStyle &st) {
Assert(rect.height() == st.size);
int index = (st.muted ? 0x03 : 0x00) + (st.active ? 0x02 : (st.selected ? 0x01 : 0x00));
int size = st.size, sizehalf = size / 2;
auto &styles = UnreadBadgeStyles();
auto badgeData = styles.sizes;
if (st.sizeId > UnreadBadgeSize()) {
Assert(st.sizeId < UnreadBadgeSize::kCount);
badgeData = &styles.sizes[static_cast<int>(st.sizeId)];
}
const auto bg = (st.sizeId == UnreadBadgeSize::ReactionInDialogs)
? styles.reactionBg[index]
: styles.bg[index];
if (badgeData->left[index].isNull()) {
const auto ratio = style::DevicePixelRatio();
int imgsize = size * ratio, imgsizehalf = sizehalf * ratio;
CreateCircleMask(badgeData, size);
badgeData->left[index] = PixmapFromImage(
ColorizeCircleHalf(badgeData, imgsize, imgsizehalf, 0, bg));
badgeData->right[index] = PixmapFromImage(ColorizeCircleHalf(
badgeData,
imgsize,
imgsizehalf,
imgsize - imgsizehalf,
bg));
}
int bar = rect.width() - 2 * sizehalf;
p.drawPixmap(rect.x(), rect.y(), badgeData->left[index]);
if (bar) {
p.fillRect(rect.x() + sizehalf, rect.y(), bar, rect.height(), bg);
}
p.drawPixmap(rect.x() + sizehalf + bar, rect.y(), badgeData->right[index]);
}
} // namespace
UnreadBadgeStyle::UnreadBadgeStyle()
: size(st::dialogsUnreadHeight)
, padding(st::dialogsUnreadPadding)
, font(st::dialogsUnreadFont) {
}
QSize CountUnreadBadgeSize(
const QString &unreadCount,
const UnreadBadgeStyle &st,
int allowDigits) {
const auto text = ComputeUnreadBadgeText(unreadCount, allowDigits);
const auto unreadRectHeight = st.size;
const auto unreadWidth = st.font->width(text);
return {
std::max(unreadWidth + 2 * st.padding, unreadRectHeight),
unreadRectHeight,
};
}
QRect PaintUnreadBadge(
QPainter &p,
const QString &unreadCount,
int x,
int y,
const UnreadBadgeStyle &st,
int allowDigits) {
const auto text = ComputeUnreadBadgeText(unreadCount, allowDigits);
const auto unreadRectHeight = st.size;
const auto unreadWidth = st.font->width(text);
const auto unreadRectWidth = std::max(
unreadWidth + 2 * st.padding,
unreadRectHeight);
const auto unreadRectLeft = ((st.align & Qt::AlignHorizontal_Mask) & style::al_center)
? (x - unreadRectWidth) / 2
: ((st.align & Qt::AlignHorizontal_Mask) & style::al_right)
? (x - unreadRectWidth)
: x;
const auto unreadRectTop = y;
const auto badge = QRect(unreadRectLeft, unreadRectTop, unreadRectWidth, unreadRectHeight);
PaintUnreadBadge(p, badge, st);
const auto textTop = st.textTop ? st.textTop : (unreadRectHeight - st.font->height) / 2;
p.setFont(st.font);
p.setPen(st.active
? st::dialogsUnreadFgActive
: st.selected
? st::dialogsUnreadFgOver
: st::dialogsUnreadFg);
p.drawText(unreadRectLeft + (unreadRectWidth - unreadWidth) / 2, unreadRectTop + textTop + st.font->ascent, text);
return badge;
}
} // namespace Ui