-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathroundedimageprovider.cpp
More file actions
129 lines (104 loc) · 5.17 KB
/
roundedimageprovider.cpp
File metadata and controls
129 lines (104 loc) · 5.17 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#include "roundedimageprovider.h"
#include <QPainter>
#include <QPainterPath>
#include <QUrl>
#include <QNetworkAccessManager>
#include <QNetworkRequest>
#include <QNetworkReply>
#include <QEventLoop>
#include <QDebug>
#include <QMutex>
// Глобальный мьютекс для сериализации сетевых запросов.
// Спасает от падений OpenSSL при попытке одновременного скачивания по HTTPS.
static QMutex s_networkMutex;
RoundedImageProvider::RoundedImageProvider()
: QDeclarativeImageProvider(QDeclarativeImageProvider::Image)
{
// Выделяем 10 МБ памяти под кэш изображений
m_cache.setMaxCost(4 * 1024 * 1024);
}
QImage RoundedImageProvider::requestImage(const QString &id, QSize *size, const QSize &requestedSize)
{
QString decodedId = QUrl::fromPercentEncoding(id.toUtf8());
// 1. БЕЗОПАСНО проверяем кэш
m_mutex.lock();
if (QImage *cachedImg = m_cache.object(decodedId)) {
QImage result = *cachedImg;
if (size) *size = result.size();
m_mutex.unlock();
return result;
}
m_mutex.unlock();
QImage originalImage;
// 2. Скачивание изображения (в фоновом потоке от QML)
if (decodedId.startsWith("http://") || decodedId.startsWith("https://")) {
// ВАЖНО: Блокируем остальные потоки.
// Пока скачивается одна картинка, другие ждут в очереди, не ломая OpenSSL.
s_networkMutex.lock();
// Создаем менеджер на стеке. Он будет безопасно уничтожен в конце блока.
QNetworkAccessManager manager;
manager.setParent(0);
QUrl requestUrl = QUrl::fromEncoded(decodedId.toUtf8());
QNetworkReply *reply = manager.get(QNetworkRequest(requestUrl));
reply->ignoreSslErrors();
// Ждем завершения скачивания именно этой картинки
QEventLoop loop;
QObject::connect(reply, SIGNAL(finished()), &loop, SLOT(quit()));
loop.exec();
if (reply->error() == QNetworkReply::NoError) {
originalImage.loadFromData(reply->readAll());
}
// ЖЕСТКОЕ удаление: отключаем сигналы и удаляем объект ответа
// Это предотвращает накопление "мусора" в памяти потока при быстром скролле
reply->disconnect();
delete reply;
s_networkMutex.unlock(); // Освобождаем очередь для следующей картинки
} else if (decodedId.startsWith("qrc:/")) {
originalImage.load(":" + decodedId.mid(4));
} else {
originalImage.load(decodedId);
}
// Заглушка: если картинка не загрузилась (нет сети), возвращаем прозрачный фон
if (originalImage.isNull()) {
int w = requestedSize.isValid() ? requestedSize.width() : 64;
int h = requestedSize.isValid() ? requestedSize.height() : 64;
QImage empty(w, h, QImage::Format_ARGB32_Premultiplied);
empty.fill(Qt::transparent);
return empty;
}
// 3. Скругляем (сглаживание Antialiasing применяется внутри roundImage)
QImage rounded = roundImage(originalImage);
// Масштабируем до требуемого размера (запрошенного QML)
if (requestedSize.isValid()) {
rounded = rounded.scaled(requestedSize, Qt::KeepAspectRatio, Qt::SmoothTransformation);
}
if (size) *size = rounded.size();
// 4. БЕЗОПАСНО сохраняем готовую круглую картинку в кэш
m_mutex.lock();
m_cache.insert(decodedId, new QImage(rounded), rounded.byteCount());
m_mutex.unlock();
return rounded;
}
QImage RoundedImageProvider::roundImage(const QImage &img)
{
// Обрезаем по центру, чтобы картинка была квадратной
int squareSize = qMin(img.width(), img.height());
QRect targetRect(0, 0, squareSize, squareSize);
QImage squareImg = img.copy((img.width() - squareSize) / 2,
(img.height() - squareSize) / 2,
squareSize, squareSize);
// Создаем пустой холст с прозрачным фоном (Alpha Channel)
QImage roundedImg(squareSize, squareSize, QImage::Format_ARGB32_Premultiplied);
roundedImg.fill(Qt::transparent);
QPainter painter(&roundedImg);
// Включаем самое качественное сглаживание краев
painter.setRenderHint(QPainter::Antialiasing, true);
painter.setRenderHint(QPainter::SmoothPixmapTransform, true);
// Накладываем круглую маску
QPainterPath path;
path.addEllipse(targetRect);
painter.setClipPath(path);
// Рисуем картинку
painter.drawImage(targetRect, squareImg);
return roundedImg;
}