2019-05-19 20:23:18 +08:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 Ivan Romanov <drizt72@zoho.eu>
|
|
|
|
*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "dialog.h"
|
|
|
|
#include <ui_dialog.h>
|
|
|
|
|
|
|
|
#include <QBuffer>
|
|
|
|
#include <QDataStream>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QFile>
|
|
|
|
#include <QFileDialog>
|
|
|
|
|
|
|
|
Dialog::Dialog(QWidget *parent)
|
|
|
|
: QDialog(parent)
|
|
|
|
, ui(new Ui::Dialog)
|
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
|
|
|
|
|
|
|
connect(ui->lwCursors, &QListWidget::currentTextChanged, this, &Dialog::showCursor);
|
|
|
|
connect(ui->pbOpenFolder, &QPushButton::clicked, this, &Dialog::openFolder);
|
|
|
|
}
|
|
|
|
|
|
|
|
Dialog::~Dialog()
|
|
|
|
{
|
|
|
|
delete ui;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Dialog::openFolder()
|
|
|
|
{
|
|
|
|
QString dirPath = QFileDialog::getExistingDirectory(this);
|
|
|
|
|
|
|
|
if (dirPath.isEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDir dir(dirPath);
|
|
|
|
|
|
|
|
QStringList fileList = dir.entryList(QDir::Filter::Files);
|
|
|
|
|
|
|
|
_cursorFileList.clear();
|
|
|
|
|
|
|
|
for (const QString &fileName: fileList) {
|
|
|
|
QFile file(dir.absoluteFilePath(fileName));
|
|
|
|
if (!file.open(QIODevice::OpenModeFlag::ReadOnly)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDataStream stream(&file);
|
|
|
|
stream.setByteOrder(QDataStream::ByteOrder::LittleEndian);
|
|
|
|
|
|
|
|
quint32 magic;
|
|
|
|
quint32 header;
|
|
|
|
quint32 version;
|
|
|
|
quint32 ntoc;
|
|
|
|
|
|
|
|
stream >> magic >> header >> version >> ntoc;
|
|
|
|
|
|
|
|
if (magic != 0x72756358 /* Xcur */ && header != 16) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
CursorFile cursorFile;
|
|
|
|
|
|
|
|
cursorFile.name = fileName;
|
|
|
|
|
|
|
|
for (quint32 i = 0; i < ntoc; ++i) {
|
|
|
|
quint32 type;
|
|
|
|
quint32 subtype;
|
|
|
|
quint32 position;
|
|
|
|
|
|
|
|
stream >> type >> subtype >> position;
|
|
|
|
qint64 tocPos = file.pos(); // position in table of contents entries
|
|
|
|
|
|
|
|
if (type == 0xfffd0002) {
|
|
|
|
file.seek(position);
|
|
|
|
|
|
|
|
quint32 imgHeader;
|
|
|
|
quint32 imgType;
|
|
|
|
quint32 imgSubtype;
|
|
|
|
quint32 imgVersion;
|
|
|
|
quint32 imgWidth;
|
|
|
|
quint32 imgHeight;
|
|
|
|
quint32 imgYhot;
|
|
|
|
quint32 imgXhot;
|
|
|
|
quint32 imgDelay;
|
|
|
|
|
|
|
|
stream >> imgHeader
|
|
|
|
>> imgType
|
|
|
|
>> imgSubtype
|
|
|
|
>> imgVersion
|
|
|
|
>> imgWidth
|
|
|
|
>> imgHeight
|
|
|
|
>> imgYhot
|
|
|
|
>> imgXhot
|
|
|
|
>> imgDelay;
|
|
|
|
|
|
|
|
if (imgHeader != 36 || imgType != type || imgSubtype != subtype || imgVersion != 1) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray imgData = file.read((imgWidth * imgHeight) * 4);
|
|
|
|
|
|
|
|
Cursor cursor;
|
|
|
|
cursor.image = QImage(reinterpret_cast<uchar*>(imgData.data()), static_cast<int>(imgWidth), static_cast<int>(imgHeight), QImage::Format::Format_ARGB32).copy();
|
|
|
|
cursor.hotSpot = QPoint(static_cast<int>(imgXhot), static_cast<int>(imgYhot));
|
2019-05-20 01:29:12 +08:00
|
|
|
cursor.size = imgSubtype;
|
2019-05-19 20:23:18 +08:00
|
|
|
|
|
|
|
QString key = QStringLiteral("%1").arg(static_cast<int>(subtype), 3, 10, QLatin1Char('0'));
|
|
|
|
cursorFile.cursorMap.insertMulti(key, cursor);
|
|
|
|
|
2019-05-20 01:29:12 +08:00
|
|
|
file.seek(tocPos);
|
|
|
|
}
|
|
|
|
else if (type == 0xfffe0001) {
|
|
|
|
file.seek(position);
|
|
|
|
|
|
|
|
quint32 commHeader;
|
|
|
|
quint32 commType;
|
|
|
|
quint32 commSubtype;
|
|
|
|
quint32 commVersion;
|
|
|
|
quint32 commLength;
|
|
|
|
|
|
|
|
stream >> commHeader
|
|
|
|
>> commType
|
|
|
|
>> commSubtype
|
|
|
|
>> commVersion
|
|
|
|
>> commLength;
|
|
|
|
|
|
|
|
if (commHeader != 20 || commType != type || commSubtype != subtype || commVersion != 1) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray commData;
|
|
|
|
commData.resize(static_cast<int>(commLength));
|
|
|
|
stream.readRawData(commData.data(), static_cast<int>(commLength));
|
|
|
|
|
|
|
|
switch (subtype) {
|
|
|
|
case 1:
|
|
|
|
if (!cursorFile.copyright.isEmpty()) {
|
|
|
|
cursorFile.copyright += QStringLiteral("\n");
|
|
|
|
}
|
|
|
|
cursorFile.copyright += QString::fromUtf8(commData);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
if (!cursorFile.license.isEmpty()) {
|
|
|
|
cursorFile.license += QStringLiteral("\n");
|
|
|
|
}
|
|
|
|
cursorFile.license += QString::fromUtf8(commData);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
if (!cursorFile.other.isEmpty()) {
|
|
|
|
cursorFile.other += QStringLiteral("\n");
|
|
|
|
}
|
|
|
|
cursorFile.other += QString::fromUtf8(commData);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-05-19 20:23:18 +08:00
|
|
|
file.seek(tocPos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_cursorFileList << cursorFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
ui->lwCursors->clear();
|
|
|
|
|
|
|
|
QStringList nameList;
|
|
|
|
|
|
|
|
for (const CursorFile &cursorFile: _cursorFileList) {
|
|
|
|
nameList << cursorFile.name;
|
|
|
|
}
|
|
|
|
|
|
|
|
ui->lwCursors->addItems(nameList);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Dialog::showCursor(const QString &fileName)
|
|
|
|
{
|
|
|
|
CursorFile foundCursorFile;
|
|
|
|
|
|
|
|
for (const CursorFile &cursorFile: _cursorFileList) {
|
|
|
|
if (cursorFile.name == fileName) {
|
|
|
|
foundCursorFile = cursorFile;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(foundCursorFile.name.isEmpty()) {
|
|
|
|
ui->teCursorInfo->clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString msg = "<html><body>";
|
|
|
|
|
|
|
|
QStringList keys = foundCursorFile.cursorMap.keys();
|
|
|
|
keys.removeDuplicates();
|
|
|
|
keys.sort();
|
|
|
|
|
2019-05-20 01:29:12 +08:00
|
|
|
if (!foundCursorFile.copyright.isEmpty()) {
|
|
|
|
msg += QStringLiteral("Copyright: %1<br/>").arg(foundCursorFile.copyright);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!foundCursorFile.license.isEmpty()) {
|
|
|
|
msg += QStringLiteral("License: %1<br/>").arg(foundCursorFile.license);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!foundCursorFile.other.isEmpty()) {
|
|
|
|
msg += QStringLiteral("Other: %1<br/>").arg(foundCursorFile.other);
|
|
|
|
}
|
|
|
|
|
2019-05-19 20:23:18 +08:00
|
|
|
for (const QString &key: keys) {
|
|
|
|
QList<Cursor> cursorList = foundCursorFile.cursorMap.values(key);
|
|
|
|
msg += "<p>";
|
2019-05-20 01:29:12 +08:00
|
|
|
Cursor firstCursor = cursorList.first();
|
|
|
|
msg += QStringLiteral("Nominal size: %1. Image size: %2x%3. Hot spot: %4x%5<br/>").arg(QString::number(firstCursor.size),
|
|
|
|
QString::number(firstCursor.image.width()), QString::number(firstCursor.image.height()),
|
|
|
|
QString::number(firstCursor.hotSpot.x()), QString::number(firstCursor.hotSpot.y()));
|
2019-05-19 20:23:18 +08:00
|
|
|
for (const Cursor &cursor: cursorList) {
|
|
|
|
QByteArray imgBa;
|
|
|
|
QBuffer buffer(&imgBa);
|
|
|
|
cursor.image.save(&buffer, "PNG");
|
|
|
|
|
|
|
|
imgBa = imgBa.toBase64();
|
|
|
|
|
|
|
|
msg += "<img src=\"data:image/png;base64, " + QString::fromLatin1(imgBa) + "\"/>";
|
|
|
|
}
|
|
|
|
msg += "</p>";
|
|
|
|
}
|
|
|
|
|
|
|
|
msg += "</body></html>";
|
|
|
|
|
|
|
|
ui->teCursorInfo->setHtml(msg);
|
|
|
|
}
|