mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-24 17:09:06 +00:00 
			
		
		
		
	New dolphin-tool command: "header" -b / --block_size -c / --compression -l / --compression_level Informative RVZ/WIA header2 value "compression_level" is now a s32 instead of a u32, because negative compression is a thing. Speaking of, it is now possible to use negative compression levels in dolphin-tool's convert command (not the GUI, though).
		
			
				
	
	
		
			517 lines
		
	
	
	
		
			18 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			517 lines
		
	
	
	
		
			18 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2020 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "DolphinQt/ConvertDialog.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <functional>
 | |
| #include <future>
 | |
| #include <memory>
 | |
| #include <utility>
 | |
| 
 | |
| #include <QCheckBox>
 | |
| #include <QComboBox>
 | |
| #include <QGridLayout>
 | |
| #include <QGroupBox>
 | |
| #include <QLabel>
 | |
| #include <QList>
 | |
| #include <QMessageBox>
 | |
| #include <QPushButton>
 | |
| #include <QString>
 | |
| #include <QVBoxLayout>
 | |
| 
 | |
| #include "Common/Assert.h"
 | |
| #include "Common/Logging/Log.h"
 | |
| #include "DiscIO/Blob.h"
 | |
| #include "DiscIO/DiscUtils.h"
 | |
| #include "DiscIO/ScrubbedBlob.h"
 | |
| #include "DiscIO/WIABlob.h"
 | |
| #include "DolphinQt/QtUtils/DolphinFileDialog.h"
 | |
| #include "DolphinQt/QtUtils/ModalMessageBox.h"
 | |
| #include "DolphinQt/QtUtils/ParallelProgressDialog.h"
 | |
| #include "UICommon/GameFile.h"
 | |
| #include "UICommon/UICommon.h"
 | |
| 
 | |
| ConvertDialog::ConvertDialog(QList<std::shared_ptr<const UICommon::GameFile>> files,
 | |
|                              QWidget* parent)
 | |
|     : QDialog(parent), m_files(std::move(files))
 | |
| {
 | |
|   ASSERT(!m_files.empty());
 | |
| 
 | |
|   setWindowTitle(tr("Convert"));
 | |
|   setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
 | |
| 
 | |
|   QGridLayout* grid_layout = new QGridLayout;
 | |
|   grid_layout->setColumnStretch(1, 1);
 | |
| 
 | |
|   m_format = new QComboBox;
 | |
|   m_format->addItem(QStringLiteral("ISO"), static_cast<int>(DiscIO::BlobType::PLAIN));
 | |
|   m_format->addItem(QStringLiteral("GCZ"), static_cast<int>(DiscIO::BlobType::GCZ));
 | |
|   m_format->addItem(QStringLiteral("WIA"), static_cast<int>(DiscIO::BlobType::WIA));
 | |
|   m_format->addItem(QStringLiteral("RVZ"), static_cast<int>(DiscIO::BlobType::RVZ));
 | |
|   if (std::all_of(m_files.begin(), m_files.end(),
 | |
|                   [](const auto& file) { return file->GetBlobType() == DiscIO::BlobType::PLAIN; }))
 | |
|   {
 | |
|     m_format->setCurrentIndex(m_format->count() - 1);
 | |
|   }
 | |
|   grid_layout->addWidget(new QLabel(tr("Format:")), 0, 0);
 | |
|   grid_layout->addWidget(m_format, 0, 1);
 | |
| 
 | |
|   m_block_size = new QComboBox;
 | |
|   grid_layout->addWidget(new QLabel(tr("Block Size:")), 1, 0);
 | |
|   grid_layout->addWidget(m_block_size, 1, 1);
 | |
| 
 | |
|   m_compression = new QComboBox;
 | |
|   grid_layout->addWidget(new QLabel(tr("Compression:")), 2, 0);
 | |
|   grid_layout->addWidget(m_compression, 2, 1);
 | |
| 
 | |
|   m_compression_level = new QComboBox;
 | |
|   grid_layout->addWidget(new QLabel(tr("Compression Level:")), 3, 0);
 | |
|   grid_layout->addWidget(m_compression_level, 3, 1);
 | |
| 
 | |
|   m_scrub = new QCheckBox;
 | |
|   grid_layout->addWidget(new QLabel(tr("Remove Junk Data (Irreversible):")), 4, 0);
 | |
|   grid_layout->addWidget(m_scrub, 4, 1);
 | |
| 
 | |
|   QPushButton* convert_button = new QPushButton(tr("Convert..."));
 | |
| 
 | |
|   QVBoxLayout* options_layout = new QVBoxLayout;
 | |
|   options_layout->addLayout(grid_layout);
 | |
|   options_layout->addWidget(convert_button);
 | |
|   QGroupBox* options_group = new QGroupBox(tr("Options"));
 | |
|   options_group->setLayout(options_layout);
 | |
| 
 | |
|   QLabel* info_text = new QLabel(
 | |
|       tr("ISO: A simple and robust format which is supported by many programs. It takes up more "
 | |
|          "space than any other format.\n\n"
 | |
|          "GCZ: A basic compressed format which is compatible with most versions of Dolphin and "
 | |
|          "some other programs. It can't efficiently compress junk data (unless removed) or "
 | |
|          "encrypted Wii data.\n\n"
 | |
|          "WIA: An advanced compressed format which is compatible with Dolphin 5.0-12188 and later, "
 | |
|          "and a few other programs. It can efficiently compress encrypted Wii data, but not junk "
 | |
|          "data (unless removed).\n\n"
 | |
|          "RVZ: An advanced compressed format which is compatible with Dolphin 5.0-12188 and later. "
 | |
|          "It can efficiently compress both junk data and encrypted Wii data."));
 | |
|   info_text->setWordWrap(true);
 | |
| 
 | |
|   QVBoxLayout* info_layout = new QVBoxLayout;
 | |
|   info_layout->addWidget(info_text);
 | |
|   QGroupBox* info_group = new QGroupBox(tr("Info"));
 | |
|   info_group->setLayout(info_layout);
 | |
| 
 | |
|   QVBoxLayout* main_layout = new QVBoxLayout;
 | |
|   main_layout->addWidget(options_group);
 | |
|   main_layout->addWidget(info_group);
 | |
| 
 | |
|   setLayout(main_layout);
 | |
| 
 | |
|   connect(m_format, QOverload<int>::of(&QComboBox::currentIndexChanged), this,
 | |
|           &ConvertDialog::OnFormatChanged);
 | |
|   connect(m_compression, QOverload<int>::of(&QComboBox::currentIndexChanged), this,
 | |
|           &ConvertDialog::OnCompressionChanged);
 | |
|   connect(convert_button, &QPushButton::clicked, this, &ConvertDialog::Convert);
 | |
| 
 | |
|   OnFormatChanged();
 | |
|   OnCompressionChanged();
 | |
| }
 | |
| 
 | |
| void ConvertDialog::AddToBlockSizeComboBox(int size)
 | |
| {
 | |
|   m_block_size->addItem(QString::fromStdString(UICommon::FormatSize(size, 0)), size);
 | |
| 
 | |
|   // Select the default, or if it is not available, the size closest to it.
 | |
|   // This code assumes that sizes get added to the combo box in increasing order.
 | |
|   if (size <= DiscIO::GCZ_RVZ_PREFERRED_BLOCK_SIZE)
 | |
|     m_block_size->setCurrentIndex(m_block_size->count() - 1);
 | |
| }
 | |
| 
 | |
| void ConvertDialog::AddToCompressionComboBox(const QString& name,
 | |
|                                              DiscIO::WIARVZCompressionType type)
 | |
| {
 | |
|   m_compression->addItem(name, static_cast<int>(type));
 | |
| }
 | |
| 
 | |
| void ConvertDialog::AddToCompressionLevelComboBox(int level)
 | |
| {
 | |
|   m_compression_level->addItem(QString::number(level), level);
 | |
| }
 | |
| 
 | |
| void ConvertDialog::OnFormatChanged()
 | |
| {
 | |
|   const DiscIO::BlobType format = static_cast<DiscIO::BlobType>(m_format->currentData().toInt());
 | |
| 
 | |
|   m_block_size->clear();
 | |
|   m_compression->clear();
 | |
| 
 | |
|   // Populate m_block_size
 | |
|   switch (format)
 | |
|   {
 | |
|   case DiscIO::BlobType::GCZ:
 | |
|   {
 | |
|     // To support legacy versions of dolphin, we have to check the GCZ block size
 | |
|     // See DiscIO::IsGCZBlockSizeLegacyCompatible() for details
 | |
|     const auto block_size_ok = [this](int block_size) {
 | |
|       return std::all_of(m_files.begin(), m_files.end(), [block_size](const auto& file) {
 | |
|         return DiscIO::IsGCZBlockSizeLegacyCompatible(block_size, file->GetVolumeSize());
 | |
|       });
 | |
|     };
 | |
| 
 | |
|     // Add all block sizes in the normal range that do not cause problems
 | |
|     for (int block_size = DiscIO::PREFERRED_MIN_BLOCK_SIZE;
 | |
|          block_size <= DiscIO::PREFERRED_MAX_BLOCK_SIZE; block_size *= 2)
 | |
|     {
 | |
|       if (block_size_ok(block_size))
 | |
|         AddToBlockSizeComboBox(block_size);
 | |
|     }
 | |
| 
 | |
|     // If we didn't find a good block size, pick the block size which was hardcoded
 | |
|     // in older versions of Dolphin. That way, at least we're not worse than older versions.
 | |
|     if (m_block_size->count() == 0)
 | |
|     {
 | |
|       if (!block_size_ok(DiscIO::GCZ_FALLBACK_BLOCK_SIZE))
 | |
|       {
 | |
|         ERROR_LOG_FMT(MASTER_LOG, "Failed to find a block size which does not cause problems "
 | |
|                                   "when decompressing using an old version of Dolphin");
 | |
|       }
 | |
|       AddToBlockSizeComboBox(DiscIO::GCZ_FALLBACK_BLOCK_SIZE);
 | |
|     }
 | |
| 
 | |
|     break;
 | |
|   }
 | |
|   case DiscIO::BlobType::WIA:
 | |
|     m_block_size->setEnabled(true);
 | |
| 
 | |
|     // This is the smallest block size supported by WIA. For performance, larger sizes are avoided.
 | |
|     AddToBlockSizeComboBox(DiscIO::WIA_MIN_BLOCK_SIZE);
 | |
| 
 | |
|     break;
 | |
|   case DiscIO::BlobType::RVZ:
 | |
|     m_block_size->setEnabled(true);
 | |
| 
 | |
|     for (int block_size = DiscIO::PREFERRED_MIN_BLOCK_SIZE;
 | |
|          block_size <= DiscIO::PREFERRED_MAX_BLOCK_SIZE; block_size *= 2)
 | |
|       AddToBlockSizeComboBox(block_size);
 | |
| 
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   // Populate m_compression
 | |
|   switch (format)
 | |
|   {
 | |
|   case DiscIO::BlobType::GCZ:
 | |
|     m_compression->setEnabled(true);
 | |
|     AddToCompressionComboBox(QStringLiteral("Deflate"), DiscIO::WIARVZCompressionType::None);
 | |
|     break;
 | |
|   case DiscIO::BlobType::WIA:
 | |
|   case DiscIO::BlobType::RVZ:
 | |
|   {
 | |
|     m_compression->setEnabled(true);
 | |
| 
 | |
|     // i18n: %1 is the name of a compression method (e.g. LZMA)
 | |
|     const QString slow = tr("%1 (slow)");
 | |
| 
 | |
|     AddToCompressionComboBox(tr("No Compression"), DiscIO::WIARVZCompressionType::None);
 | |
| 
 | |
|     if (format == DiscIO::BlobType::WIA)
 | |
|       AddToCompressionComboBox(QStringLiteral("Purge"), DiscIO::WIARVZCompressionType::Purge);
 | |
| 
 | |
|     AddToCompressionComboBox(slow.arg(QStringLiteral("bzip2")),
 | |
|                              DiscIO::WIARVZCompressionType::Bzip2);
 | |
| 
 | |
|     AddToCompressionComboBox(slow.arg(QStringLiteral("LZMA")), DiscIO::WIARVZCompressionType::LZMA);
 | |
| 
 | |
|     AddToCompressionComboBox(slow.arg(QStringLiteral("LZMA2")),
 | |
|                              DiscIO::WIARVZCompressionType::LZMA2);
 | |
| 
 | |
|     if (format == DiscIO::BlobType::RVZ)
 | |
|     {
 | |
|       // i18n: %1 is the name of a compression method (e.g. Zstandard)
 | |
|       const QString recommended = tr("%1 (recommended)");
 | |
| 
 | |
|       AddToCompressionComboBox(recommended.arg(QStringLiteral("Zstandard")),
 | |
|                                DiscIO::WIARVZCompressionType::Zstd);
 | |
|       m_compression->setCurrentIndex(m_compression->count() - 1);
 | |
|     }
 | |
| 
 | |
|     break;
 | |
|   }
 | |
|   default:
 | |
|     m_compression->setEnabled(false);
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   m_block_size->setEnabled(m_block_size->count() > 1);
 | |
|   m_compression->setEnabled(m_compression->count() > 1);
 | |
| 
 | |
|   // Block scrubbing of RVZ containers and Datel discs
 | |
|   const bool scrubbing_allowed =
 | |
|       format != DiscIO::BlobType::RVZ &&
 | |
|       std::none_of(m_files.begin(), m_files.end(), std::mem_fn(&UICommon::GameFile::IsDatelDisc));
 | |
| 
 | |
|   m_scrub->setEnabled(scrubbing_allowed);
 | |
|   if (!scrubbing_allowed)
 | |
|     m_scrub->setChecked(false);
 | |
| }
 | |
| 
 | |
| void ConvertDialog::OnCompressionChanged()
 | |
| {
 | |
|   m_compression_level->clear();
 | |
| 
 | |
|   const auto compression_type =
 | |
|       static_cast<DiscIO::WIARVZCompressionType>(m_compression->currentData().toInt());
 | |
| 
 | |
|   const std::pair<int, int> range = DiscIO::GetAllowedCompressionLevels(compression_type, true);
 | |
| 
 | |
|   for (int i = range.first; i <= range.second; ++i)
 | |
|   {
 | |
|     AddToCompressionLevelComboBox(i);
 | |
|     if (i == 5)
 | |
|       m_compression_level->setCurrentIndex(m_compression_level->count() - 1);
 | |
|   }
 | |
| 
 | |
|   m_compression_level->setEnabled(m_compression_level->count() > 1);
 | |
| }
 | |
| 
 | |
| bool ConvertDialog::ShowAreYouSureDialog(const QString& text)
 | |
| {
 | |
|   ModalMessageBox warning(this);
 | |
|   warning.setIcon(QMessageBox::Warning);
 | |
|   warning.setWindowTitle(tr("Confirm"));
 | |
|   warning.setText(tr("Are you sure?"));
 | |
|   warning.setInformativeText(text);
 | |
|   warning.setStandardButtons(QMessageBox::Yes | QMessageBox::No);
 | |
| 
 | |
|   return warning.exec() == QMessageBox::Yes;
 | |
| }
 | |
| 
 | |
| void ConvertDialog::Convert()
 | |
| {
 | |
|   const DiscIO::BlobType format = static_cast<DiscIO::BlobType>(m_format->currentData().toInt());
 | |
|   const int block_size = m_block_size->currentData().toInt();
 | |
|   const DiscIO::WIARVZCompressionType compression =
 | |
|       static_cast<DiscIO::WIARVZCompressionType>(m_compression->currentData().toInt());
 | |
|   const int compression_level = m_compression_level->currentData().toInt();
 | |
|   const bool scrub = m_scrub->isChecked();
 | |
| 
 | |
|   if (scrub && format == DiscIO::BlobType::PLAIN)
 | |
|   {
 | |
|     if (!ShowAreYouSureDialog(tr("Removing junk data does not save any space when converting to "
 | |
|                                  "ISO (unless you package the ISO file in a compressed file format "
 | |
|                                  "such as ZIP afterwards). Do you want to continue anyway?")))
 | |
|     {
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!scrub && format == DiscIO::BlobType::GCZ &&
 | |
|       std::any_of(m_files.begin(), m_files.end(), [](const auto& file) {
 | |
|         return file->GetPlatform() == DiscIO::Platform::WiiDisc && !file->IsDatelDisc();
 | |
|       }))
 | |
|   {
 | |
|     if (!ShowAreYouSureDialog(tr("Converting Wii disc images to GCZ without removing junk data "
 | |
|                                  "does not save any noticeable amount of space compared to "
 | |
|                                  "converting to ISO. Do you want to continue anyway?")))
 | |
|     {
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (std::any_of(m_files.begin(), m_files.end(), std::mem_fn(&UICommon::GameFile::IsNKit)))
 | |
|   {
 | |
|     if (!ShowAreYouSureDialog(
 | |
|             tr("Dolphin can't convert NKit files to non-NKit files. Converting an NKit file in "
 | |
|                "Dolphin will result in another NKit file.\n"
 | |
|                "\n"
 | |
|                "If you want to convert an NKit file to a non-NKit file, you can use the same "
 | |
|                "program as you originally used when converting the file to the NKit format.\n"
 | |
|                "\n"
 | |
|                "Do you want to continue anyway?")))
 | |
|     {
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   QString extension;
 | |
|   QString filter;
 | |
|   switch (format)
 | |
|   {
 | |
|   case DiscIO::BlobType::PLAIN:
 | |
|     extension = QStringLiteral(".iso");
 | |
|     filter = tr("Uncompressed GC/Wii images (*.iso *.gcm)");
 | |
|     break;
 | |
|   case DiscIO::BlobType::GCZ:
 | |
|     extension = QStringLiteral(".gcz");
 | |
|     filter = tr("GCZ GC/Wii images (*.gcz)");
 | |
|     break;
 | |
|   case DiscIO::BlobType::WIA:
 | |
|     extension = QStringLiteral(".wia");
 | |
|     filter = tr("WIA GC/Wii images (*.wia)");
 | |
|     break;
 | |
|   case DiscIO::BlobType::RVZ:
 | |
|     extension = QStringLiteral(".rvz");
 | |
|     filter = tr("RVZ GC/Wii images (*.rvz)");
 | |
|     break;
 | |
|   default:
 | |
|     ASSERT(false);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   QString dst_dir;
 | |
|   QString dst_path;
 | |
| 
 | |
|   if (m_files.size() > 1)
 | |
|   {
 | |
|     dst_dir = DolphinFileDialog::getExistingDirectory(
 | |
|         this, tr("Select where you want to save the converted images"),
 | |
|         QFileInfo(QString::fromStdString(m_files[0]->GetFilePath())).dir().absolutePath());
 | |
| 
 | |
|     if (dst_dir.isEmpty())
 | |
|       return;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     dst_path = DolphinFileDialog::getSaveFileName(
 | |
|         this, tr("Select where you want to save the converted image"),
 | |
|         QFileInfo(QString::fromStdString(m_files[0]->GetFilePath()))
 | |
|             .dir()
 | |
|             .absoluteFilePath(
 | |
|                 QFileInfo(QString::fromStdString(m_files[0]->GetFilePath())).completeBaseName())
 | |
|             .append(extension),
 | |
|         filter);
 | |
| 
 | |
|     if (dst_path.isEmpty())
 | |
|       return;
 | |
|   }
 | |
| 
 | |
|   for (const auto& file : m_files)
 | |
|   {
 | |
|     const auto original_path = file->GetFilePath();
 | |
|     if (m_files.size() > 1)
 | |
|     {
 | |
|       dst_path =
 | |
|           QDir(dst_dir)
 | |
|               .absoluteFilePath(QFileInfo(QString::fromStdString(original_path)).completeBaseName())
 | |
|               .append(extension);
 | |
|       QFileInfo dst_info = QFileInfo(dst_path);
 | |
|       if (dst_info.exists())
 | |
|       {
 | |
|         ModalMessageBox confirm_replace(this);
 | |
|         confirm_replace.setIcon(QMessageBox::Warning);
 | |
|         confirm_replace.setWindowTitle(tr("Confirm"));
 | |
|         confirm_replace.setText(tr("The file %1 already exists.\n"
 | |
|                                    "Do you wish to replace it?")
 | |
|                                     .arg(dst_info.fileName()));
 | |
|         confirm_replace.setStandardButtons(QMessageBox::Yes | QMessageBox::No);
 | |
| 
 | |
|         if (confirm_replace.exec() == QMessageBox::No)
 | |
|           continue;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     ParallelProgressDialog progress_dialog(tr("Converting..."), tr("Abort"), 0, 100, this);
 | |
|     progress_dialog.GetRaw()->setWindowModality(Qt::WindowModal);
 | |
|     progress_dialog.GetRaw()->setWindowTitle(tr("Progress"));
 | |
| 
 | |
|     if (m_files.size() > 1)
 | |
|     {
 | |
|       // i18n: %1 is a filename.
 | |
|       progress_dialog.GetRaw()->setLabelText(
 | |
|           tr("Converting...\n%1").arg(QFileInfo(QString::fromStdString(original_path)).fileName()));
 | |
|     }
 | |
| 
 | |
|     std::unique_ptr<DiscIO::BlobReader> blob_reader;
 | |
|     bool scrub_current_file = scrub;
 | |
| 
 | |
|     if (scrub_current_file)
 | |
|     {
 | |
|       blob_reader = DiscIO::ScrubbedBlob::Create(original_path);
 | |
|       if (!blob_reader)
 | |
|       {
 | |
|         const int result =
 | |
|             ModalMessageBox::warning(this, tr("Question"),
 | |
|                                      tr("Failed to remove junk data from file \"%1\".\n\n"
 | |
|                                         "Would you like to convert it without removing junk data?")
 | |
|                                          .arg(QString::fromStdString(original_path)),
 | |
|                                      QMessageBox::Ok | QMessageBox::Abort);
 | |
| 
 | |
|         if (result == QMessageBox::Ok)
 | |
|           scrub_current_file = false;
 | |
|         else
 | |
|           return;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (!scrub_current_file)
 | |
|       blob_reader = DiscIO::CreateBlobReader(original_path);
 | |
| 
 | |
|     if (!blob_reader)
 | |
|     {
 | |
|       ModalMessageBox::critical(
 | |
|           this, tr("Error"),
 | |
|           tr("Failed to open the input file \"%1\".").arg(QString::fromStdString(original_path)));
 | |
|       return;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|       const auto callback = [&progress_dialog](const std::string& text, float percent) {
 | |
|         progress_dialog.SetValue(percent * 100);
 | |
|         return !progress_dialog.WasCanceled();
 | |
|       };
 | |
| 
 | |
|       std::future<bool> success;
 | |
| 
 | |
|       switch (format)
 | |
|       {
 | |
|       case DiscIO::BlobType::PLAIN:
 | |
|         success = std::async(std::launch::async, [&] {
 | |
|           const bool good = DiscIO::ConvertToPlain(blob_reader.get(), original_path,
 | |
|                                                    dst_path.toStdString(), callback);
 | |
|           progress_dialog.Reset();
 | |
|           return good;
 | |
|         });
 | |
|         break;
 | |
| 
 | |
|       case DiscIO::BlobType::GCZ:
 | |
|         success = std::async(std::launch::async, [&] {
 | |
|           const bool good = DiscIO::ConvertToGCZ(
 | |
|               blob_reader.get(), original_path, dst_path.toStdString(),
 | |
|               file->GetPlatform() == DiscIO::Platform::WiiDisc ? 1 : 0, block_size, callback);
 | |
|           progress_dialog.Reset();
 | |
|           return good;
 | |
|         });
 | |
|         break;
 | |
| 
 | |
|       case DiscIO::BlobType::WIA:
 | |
|       case DiscIO::BlobType::RVZ:
 | |
|         success = std::async(std::launch::async, [&] {
 | |
|           const bool good =
 | |
|               DiscIO::ConvertToWIAOrRVZ(blob_reader.get(), original_path, dst_path.toStdString(),
 | |
|                                         format == DiscIO::BlobType::RVZ, compression,
 | |
|                                         compression_level, block_size, callback);
 | |
|           progress_dialog.Reset();
 | |
|           return good;
 | |
|         });
 | |
|         break;
 | |
| 
 | |
|       default:
 | |
|         ASSERT(false);
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       progress_dialog.GetRaw()->exec();
 | |
|       if (!success.get())
 | |
|       {
 | |
|         ModalMessageBox::critical(this, tr("Error"),
 | |
|                                   tr("Dolphin failed to complete the requested action."));
 | |
|         return;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   ModalMessageBox::information(this, tr("Success"),
 | |
|                                tr("Successfully converted %n image(s).", "", m_files.size()));
 | |
| 
 | |
|   close();
 | |
| }
 |