mirror of https://github.com/PCSX2/pcsx2.git
Qt: Add dump verification to game properties
This commit is contained in:
parent
73f903f402
commit
2539a07b7d
|
@ -19,17 +19,24 @@
|
|||
#include "SettingsDialog.h"
|
||||
#include "MainWindow.h"
|
||||
#include "QtHost.h"
|
||||
#include "QtProgressCallback.h"
|
||||
#include "QtUtils.h"
|
||||
|
||||
#include "pcsx2/CDVD/IsoHasher.h"
|
||||
#include "pcsx2/GameDatabase.h"
|
||||
#include "pcsx2/GameList.h"
|
||||
#include "pcsx2/PAD/Host/PAD.h"
|
||||
|
||||
#include "common/Error.h"
|
||||
#include "common/MD5Digest.h"
|
||||
#include "common/ScopedGuard.h"
|
||||
#include "common/StringUtil.h"
|
||||
|
||||
#include "fmt/format.h"
|
||||
|
||||
#include <QtCore/QDir>
|
||||
#include <QtWidgets/QFileDialog>
|
||||
#include <QtWidgets/QMessageBox>
|
||||
|
||||
GameSummaryWidget::GameSummaryWidget(const GameList::Entry* entry, SettingsDialog* dialog, QWidget* parent)
|
||||
: m_dialog(dialog)
|
||||
|
@ -51,8 +58,11 @@ GameSummaryWidget::GameSummaryWidget(const GameList::Entry* entry, SettingsDialo
|
|||
populateInputProfiles();
|
||||
populateDetails(entry);
|
||||
populateDiscPath(entry);
|
||||
populateTrackList(entry);
|
||||
|
||||
connect(m_ui.inputProfile, &QComboBox::currentIndexChanged, this, &GameSummaryWidget::onInputProfileChanged);
|
||||
connect(m_ui.verify, &QAbstractButton::clicked, this, &GameSummaryWidget::onVerifyClicked);
|
||||
connect(m_ui.searchHash, &QAbstractButton::clicked, this, &GameSummaryWidget::onSearchHashClicked);
|
||||
}
|
||||
|
||||
GameSummaryWidget::~GameSummaryWidget() = default;
|
||||
|
@ -137,3 +147,209 @@ void GameSummaryWidget::onDiscPathBrowseClicked()
|
|||
// let the signal take care of it
|
||||
m_ui.discPath->setText(QDir::toNativeSeparators(filename));
|
||||
}
|
||||
|
||||
void GameSummaryWidget::populateTrackList(const GameList::Entry* entry)
|
||||
{
|
||||
if (entry->type != GameList::EntryType::PS1Disc && entry->type != GameList::EntryType::PS2Disc)
|
||||
{
|
||||
m_ui.verify->setEnabled(false);
|
||||
m_ui.verifyResult->setPlainText(tr("Game is not a CD/DVD."));
|
||||
return;
|
||||
}
|
||||
|
||||
if (QtHost::IsVMValid())
|
||||
{
|
||||
m_ui.verify->setEnabled(false);
|
||||
m_ui.verifyResult->setPlainText(tr("Track list unavailable while virtual machine is running."));
|
||||
return;
|
||||
}
|
||||
|
||||
IsoHasher hasher;
|
||||
Error error;
|
||||
if (!hasher.Open(m_entry_path, &error))
|
||||
{
|
||||
m_ui.verify->setEnabled(false);
|
||||
m_ui.verifyResult->setPlainText(QString::fromStdString(error.GetDescription()));
|
||||
return;
|
||||
}
|
||||
|
||||
const auto AddColumn = [this](const QString& text) {
|
||||
QTableWidgetItem* item = new QTableWidgetItem(text);
|
||||
const int column = m_ui.tracks->columnCount();
|
||||
m_ui.tracks->insertColumn(column);
|
||||
m_ui.tracks->setHorizontalHeaderItem(column, item);
|
||||
};
|
||||
const auto SetColumn = [this](int row, int column, const QString& text) {
|
||||
QTableWidgetItem* item = new QTableWidgetItem(text);
|
||||
m_ui.tracks->setItem(row, column, item);
|
||||
};
|
||||
|
||||
// columns depend on CD vs DVD.
|
||||
AddColumn(tr("#"));
|
||||
if (hasher.IsCD())
|
||||
{
|
||||
AddColumn(tr("Mode"));
|
||||
AddColumn(tr("Start"));
|
||||
AddColumn(tr("Sectors"));
|
||||
AddColumn(tr("Size"));
|
||||
AddColumn(tr("MD5"));
|
||||
AddColumn(tr("Status"));
|
||||
}
|
||||
else
|
||||
{
|
||||
AddColumn(tr("Start"));
|
||||
AddColumn(tr("Sectors"));
|
||||
AddColumn(tr("Size"));
|
||||
AddColumn(tr("MD5"));
|
||||
AddColumn(tr("Status"));
|
||||
}
|
||||
|
||||
for (const IsoHasher::Track& track : hasher.GetTracks())
|
||||
{
|
||||
const int row = m_ui.tracks->rowCount();
|
||||
m_ui.tracks->insertRow(row);
|
||||
|
||||
SetColumn(row, 0, tr("%1").arg(track.number));
|
||||
|
||||
if (hasher.IsCD())
|
||||
{
|
||||
SetColumn(row, 1, QtUtils::StringViewToQString(IsoHasher::GetTrackTypeString(track.type)));
|
||||
SetColumn(row, 2, tr("%1").arg(track.start_lsn));
|
||||
SetColumn(row, 3, tr("%1").arg(track.sectors));
|
||||
SetColumn(row, 4, tr("%1").arg(track.size));
|
||||
SetColumn(row, 5, tr("<not computed>"));
|
||||
SetColumn(row, 6, QString());
|
||||
}
|
||||
else
|
||||
{
|
||||
SetColumn(row, 1, tr("%1").arg(track.start_lsn));
|
||||
SetColumn(row, 2, tr("%1").arg(track.sectors));
|
||||
SetColumn(row, 3, tr("%1").arg(track.size));
|
||||
SetColumn(row, 4, tr("<not computed>"));
|
||||
SetColumn(row, 5, QString());
|
||||
}
|
||||
}
|
||||
|
||||
if (hasher.IsCD())
|
||||
QtUtils::ResizeColumnsForTableView(m_ui.tracks, {20, 60, 70, 70, 100, 220, 40});
|
||||
else
|
||||
QtUtils::ResizeColumnsForTableView(m_ui.tracks, {20, 100, 100, 100, 220, 40});
|
||||
}
|
||||
|
||||
void GameSummaryWidget::onVerifyClicked()
|
||||
{
|
||||
// Can't do this while a VM is running because of stupid CDVD.
|
||||
if (QtHost::IsVMValid())
|
||||
{
|
||||
QMessageBox::critical(QtUtils::GetRootWidget(this), tr("Error"), tr("Cannot verify image while a game is running."));
|
||||
return;
|
||||
}
|
||||
|
||||
IsoHasher hasher;
|
||||
Error error;
|
||||
if (!hasher.Open(m_entry_path, &error))
|
||||
{
|
||||
setVerifyResult(QString::fromStdString(error.GetDescription()));
|
||||
return;
|
||||
}
|
||||
|
||||
QtModalProgressCallback callback(this);
|
||||
hasher.ComputeHashes(&callback);
|
||||
if (callback.IsCancelled())
|
||||
return;
|
||||
|
||||
const int hash_column = hasher.IsCD() ? 5 : 4;
|
||||
int row = 0;
|
||||
|
||||
// convert to database format
|
||||
std::vector<GameDatabase::TrackHash> thashes;
|
||||
thashes.reserve(hasher.GetTrackCount());
|
||||
for (const IsoHasher::Track& track : hasher.GetTracks())
|
||||
{
|
||||
GameDatabase::TrackHash thash;
|
||||
thash.size = track.size;
|
||||
if (track.hash.empty() || !thash.parseHash(track.hash))
|
||||
{
|
||||
m_ui.verify->setEnabled(false);
|
||||
m_ui.verifyResult->setPlainText(tr("One or more tracks is missing."));
|
||||
return;
|
||||
}
|
||||
|
||||
// Use the first track's hash as the redump search term.
|
||||
if (m_redump_search_keyword.empty())
|
||||
m_redump_search_keyword = thash.toString();
|
||||
|
||||
thashes.push_back(thash);
|
||||
}
|
||||
|
||||
// match the hashes. can't use vector<bool> here because it's not an actual array
|
||||
std::unique_ptr<bool[]> val_results = std::make_unique<bool[]>(hasher.GetTrackCount());
|
||||
std::string match_error;
|
||||
const GameDatabase::HashDatabaseEntry* hentry =
|
||||
GameDatabase::lookupHash(thashes.data(), thashes.size(), val_results.get(), &match_error);
|
||||
|
||||
// fill the UI with both the hashes and validation results
|
||||
for (u32 i = 0; i < hasher.GetTrackCount(); i++)
|
||||
{
|
||||
QTableWidgetItem* const hash_item = m_ui.tracks->item(row, hash_column);
|
||||
QTableWidgetItem* const status_item = m_ui.tracks->item(row, hash_column + 1);
|
||||
|
||||
const bool result = val_results[i];
|
||||
const QBrush brush(result ? QColor(0, 200, 0) : QColor(200, 0, 0));
|
||||
|
||||
hash_item->setText(QString::fromStdString(hasher.GetTrack(i).hash));
|
||||
hash_item->setForeground(brush);
|
||||
status_item->setText(result ? QStringLiteral("\u2713") : QStringLiteral("\u2715"));
|
||||
status_item->setForeground(brush);
|
||||
row++;
|
||||
}
|
||||
|
||||
if (hentry)
|
||||
{
|
||||
if (!hentry->version.empty())
|
||||
{
|
||||
setVerifyResult(tr("Verified as %1 [%2] (Version %3).")
|
||||
.arg(QString::fromStdString(hentry->name))
|
||||
.arg(QString::fromStdString(hentry->serial))
|
||||
.arg(QString::fromStdString(hentry->version)));
|
||||
}
|
||||
else
|
||||
{
|
||||
setVerifyResult(tr("Verified as %1 [%2].")
|
||||
.arg(QString::fromStdString(hentry->name))
|
||||
.arg(QString::fromStdString(hentry->serial)));
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
setVerifyResult(QString::fromStdString(match_error));
|
||||
}
|
||||
}
|
||||
|
||||
void GameSummaryWidget::onSearchHashClicked()
|
||||
{
|
||||
if (m_redump_search_keyword.empty())
|
||||
return;
|
||||
|
||||
QtUtils::OpenURL(this, fmt::format("http://redump.org/discs/quicksearch/{}", m_redump_search_keyword).c_str());
|
||||
}
|
||||
|
||||
void GameSummaryWidget::setVerifyResult(QString error)
|
||||
{
|
||||
m_ui.verify->setVisible(false);
|
||||
m_ui.verifyButtonLayout->removeWidget(m_ui.verify);
|
||||
m_ui.verify->deleteLater();
|
||||
m_ui.verify = nullptr;
|
||||
m_ui.verifyButtonLayout->removeItem(m_ui.verifyButtonSpacer);
|
||||
delete m_ui.verifyButtonSpacer;
|
||||
m_ui.verifyButtonSpacer = nullptr;
|
||||
m_ui.verifyLayout->removeItem(m_ui.verifyButtonLayout);
|
||||
m_ui.verifyButtonLayout->deleteLater();
|
||||
m_ui.verifyButtonLayout = nullptr;
|
||||
m_ui.verifyLayout->update();
|
||||
updateGeometry();
|
||||
|
||||
m_ui.verifyResult->setPlainText(error);
|
||||
m_ui.verifyResult->setVisible(true);
|
||||
m_ui.searchHash->setVisible(true);
|
||||
}
|
|
@ -34,16 +34,22 @@ public:
|
|||
GameSummaryWidget(const GameList::Entry* entry, SettingsDialog* dialog, QWidget* parent);
|
||||
~GameSummaryWidget();
|
||||
|
||||
private Q_SLOTS:
|
||||
void onInputProfileChanged(int index);
|
||||
void onDiscPathChanged(const QString& value);
|
||||
void onDiscPathBrowseClicked();
|
||||
void onVerifyClicked();
|
||||
void onSearchHashClicked();
|
||||
|
||||
private:
|
||||
void populateInputProfiles();
|
||||
void populateDetails(const GameList::Entry* entry);
|
||||
void populateDiscPath(const GameList::Entry* entry);
|
||||
|
||||
void onInputProfileChanged(int index);
|
||||
void onDiscPathChanged(const QString& value);
|
||||
void onDiscPathBrowseClicked();
|
||||
void populateTrackList(const GameList::Entry* entry);
|
||||
void setVerifyResult(QString error);
|
||||
|
||||
Ui::GameSummaryWidget m_ui;
|
||||
SettingsDialog* m_dialog;
|
||||
std::string m_entry_path;
|
||||
std::string m_redump_search_keyword;
|
||||
};
|
||||
|
|
|
@ -6,8 +6,8 @@
|
|||
<rect>
|
||||
<x>0</x>
|
||||
<y>0</y>
|
||||
<width>692</width>
|
||||
<height>562</height>
|
||||
<width>641</width>
|
||||
<height>467</height>
|
||||
</rect>
|
||||
</property>
|
||||
<layout class="QFormLayout" name="detailsFormLayout">
|
||||
|
@ -413,7 +413,7 @@
|
|||
</item>
|
||||
</layout>
|
||||
</item>
|
||||
<item row="9" column="1">
|
||||
<item row="10" column="0" colspan="2">
|
||||
<spacer name="verticalSpacer">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Vertical</enum>
|
||||
|
@ -426,6 +426,90 @@
|
|||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item row="9" column="0" colspan="2">
|
||||
<layout class="QVBoxLayout" name="verifyLayout">
|
||||
<item>
|
||||
<widget class="QTableWidget" name="tracks">
|
||||
<property name="editTriggers">
|
||||
<set>QAbstractItemView::NoEditTriggers</set>
|
||||
</property>
|
||||
<property name="cornerButtonEnabled">
|
||||
<bool>false</bool>
|
||||
</property>
|
||||
<attribute name="verticalHeaderVisible">
|
||||
<bool>false</bool>
|
||||
</attribute>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<layout class="QHBoxLayout" name="verifyButtonLayout">
|
||||
<item>
|
||||
<spacer name="verifyButtonSpacer">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QPushButton" name="verify">
|
||||
<property name="text">
|
||||
<string>Verify</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</item>
|
||||
<item>
|
||||
<layout class="QGridLayout" name="gridLayout">
|
||||
<item row="2" column="2">
|
||||
<spacer name="verticalSpacer_2">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Vertical</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>20</width>
|
||||
<height>40</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item row="0" column="0" rowspan="3">
|
||||
<widget class="QPlainTextEdit" name="verifyResult">
|
||||
<property name="maximumSize">
|
||||
<size>
|
||||
<width>16777215</width>
|
||||
<height>60</height>
|
||||
</size>
|
||||
</property>
|
||||
<property name="readOnly">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<property name="visible">
|
||||
<bool>false</bool>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="1" column="2">
|
||||
<widget class="QPushButton" name="searchHash">
|
||||
<property name="text">
|
||||
<string>Search on Redump.org...</string>
|
||||
</property>
|
||||
<property name="visible">
|
||||
<bool>false</bool>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</item>
|
||||
</layout>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<resources>
|
||||
|
|
|
@ -79,8 +79,10 @@ void SettingsDialog::setupUi(const GameList::Entry* game)
|
|||
|
||||
if (isPerGameSettings())
|
||||
{
|
||||
QString summary = tr("<strong>Summary</strong><hr>Eventually this will be where we can see patches and compute "
|
||||
"hashes/verify dumps/etc.");
|
||||
QString summary = tr("<strong>Summary</strong><hr>This page shows details about the selected game. Changing the Input "
|
||||
"Profile will set the controller binding scheme for this game to whichever profile is chosen, instead "
|
||||
"of the default (Shared) configuration. The track list and dump verification can be used to determine "
|
||||
"if your disc image matches a known good dump. If it does not match, the game may be broken.");
|
||||
if (game)
|
||||
{
|
||||
addWidget(new GameSummaryWidget(game, this, m_ui.settingsContainer), tr("Summary"),
|
||||
|
|
Loading…
Reference in New Issue