diff --git a/src/ui/ripcd.cpp b/src/ui/ripcd.cpp index 5828a4347..91637dc7f 100644 --- a/src/ui/ripcd.cpp +++ b/src/ui/ripcd.cpp @@ -49,8 +49,8 @@ #endif static bool ComparePresetsByName(const TranscoderPreset& left, - const TranscoderPreset& right) { - return left.name_ < right.name_; + const TranscoderPreset& right) { + return left.name_ < right.name_; } const char* RipCD::kSettingsGroup = "Transcoder"; @@ -58,293 +58,295 @@ const int RipCD::kProgressInterval = 500; const int RipCD::kMaxDestinationItems = 10; RipCD::RipCD(QWidget* parent) : - QDialog(parent), transcoder_(new Transcoder(this)), queued_(0), finished_success_( - 0), finished_failed_(0), ui_(new Ui_RipCD()), checkboxes_( - QList()), generated_files_(QList()), tracks_to_rip_( - QList()), track_names_(QList()) { - // Init - ui_->setupUi(this); - cancel_button_ = ui_->button_box->button(QDialogButtonBox::Cancel); + QDialog(parent), transcoder_(new Transcoder(this)), queued_(0), finished_success_( + 0), finished_failed_(0), ui_(new Ui_RipCD()), checkboxes_( + QList()), generated_files_(QList()), tracks_to_rip_( + QList()), track_names_(QList()) { + // Init + ui_->setupUi(this); + cancel_button_ = ui_->button_box->button(QDialogButtonBox::Cancel); - connect(ui_->ripButton, SIGNAL(clicked()), this, SLOT(clickedRipButton())); - connect(cancel_button_, SIGNAL(clicked()), SLOT(Cancel())); + connect(ui_->ripButton, SIGNAL(clicked()), this, SLOT(ClickedRipButton())); + connect(cancel_button_, SIGNAL(clicked()), SLOT(Cancel())); - connect(transcoder_, SIGNAL(JobComplete(QString,bool)), SLOT(JobComplete(QString,bool))); - connect(transcoder_, SIGNAL(AllJobsComplete()), SLOT(AllJobsComplete())); - connect(transcoder_, SIGNAL(JobOutputName(QString)), - SLOT(AppendOutput(QString))); - connect(this, SIGNAL(RippingComplete()), SLOT(threadedTranscoding())); - connect(this, SIGNAL(signalUpdateProgress()), SLOT(UpdateProgress())); + connect(transcoder_, SIGNAL(JobComplete(QString, bool)), SLOT(JobComplete(QString, bool))); + connect(transcoder_, SIGNAL(AllJobsComplete()), SLOT(AllJobsComplete())); + connect(transcoder_, SIGNAL(JobOutputName(QString)), + SLOT(AppendOutput(QString))); + connect(this, SIGNAL(RippingComplete()), SLOT(ThreadedTranscoding())); + connect(this, SIGNAL(signalUpdateProgress()), SLOT(UpdateProgress())); - connect(ui_->options, SIGNAL(clicked()), SLOT(Options())); - connect(ui_->select, SIGNAL(clicked()), SLOT(AddDestination())); + connect(ui_->options, SIGNAL(clicked()), SLOT(Options())); + connect(ui_->select, SIGNAL(clicked()), SLOT(AddDestination())); - setWindowTitle(tr("Rip CD")); + setWindowTitle(tr("Rip CD")); - //track_t i_first_track; - p_cdio = cdio_open(NULL, DRIVER_UNKNOWN); - //i_first_track = cdio_get_first_track_num(p_cdio); - i_tracks = cdio_get_num_tracks(p_cdio); - ui_->tableWidget->setRowCount(i_tracks); - for (int i = 1; i <= i_tracks; i++) { - QCheckBox *_t = new QCheckBox(tr(""), ui_->tableWidget); - _t->click(); - checkboxes_.append(_t); - ui_->tableWidget->setCellWidget(i - 1, 0, _t); - ui_->tableWidget->setCellWidget(i - 1, 1, new QLabel(QString::number(i))); - QString _str_track; - _str_track = "Track %1"; - QLineEdit *_ql = new QLineEdit(_str_track.arg(QString::number(i)), - ui_->tableWidget); - track_names_.append(_ql); - ui_->tableWidget->setCellWidget(i - 1, 2, _ql); - } - // Get presets - QList < TranscoderPreset > presets = Transcoder::GetAllPresets(); - qSort(presets.begin(), presets.end(), ComparePresetsByName); - foreach (const TranscoderPreset& preset, presets) { - ui_->format->addItem( - QString("%1 (.%2)").arg(preset.name_, preset.extension_), - QVariant::fromValue(preset)); - } + p_cdio_ = cdio_open(NULL, DRIVER_UNKNOWN); + i_tracks = cdio_get_num_tracks(p_cdio_); + ui_->tableWidget->setRowCount(i_tracks); + for (int i = 1; i <= i_tracks; i++) { + QCheckBox *_t = new QCheckBox(tr(""), ui_->tableWidget); + _t->click(); + checkboxes_.append(_t); + ui_->tableWidget->setCellWidget(i - 1, 0, _t); + ui_->tableWidget->setCellWidget(i - 1, 1, new QLabel(QString::number(i))); + QString _str_track; + _str_track = "Track %1"; + QLineEdit *_ql = new QLineEdit(_str_track.arg(QString::number(i)), + ui_->tableWidget); + track_names_.append(_ql); + ui_->tableWidget->setCellWidget(i - 1, 2, _ql); + } + // Get presets + QList < TranscoderPreset > presets = Transcoder::GetAllPresets(); + qSort(presets.begin(), presets.end(), ComparePresetsByName); + foreach(const TranscoderPreset& preset, presets) { + ui_->format->addItem( + QString("%1 (.%2)").arg(preset.name_, preset.extension_), + QVariant::fromValue(preset)); + } - // Load settings - QSettings s; - s.beginGroup(kSettingsGroup); - last_add_dir_ = s.value("last_add_dir", QDir::homePath()).toString(); + // Load settings + QSettings s; + s.beginGroup(kSettingsGroup); + last_add_dir_ = s.value("last_add_dir", QDir::homePath()).toString(); - QString last_output_format = s.value("last_output_format", "ogg").toString(); - for (int i = 0; i < ui_->format->count(); ++i) { - if (last_output_format - == ui_->format->itemData(i).value().extension_) { - ui_->format->setCurrentIndex(i); - break; - } - } + QString last_output_format = s.value("last_output_format", "ogg").toString(); + for (int i = 0; i < ui_->format->count(); ++i) { + if (last_output_format + == ui_->format->itemData(i).value().extension_) { + ui_->format->setCurrentIndex(i); + break; + } + } - ui_->progress_bar->setValue(0); - ui_->progress_bar->setMaximum(100); + ui_->progress_bar->setValue(0); + ui_->progress_bar->setMaximum(100); } -void RipCD::write_WAV_header(FILE *stream, int32_t i_bytecount) { - fwrite("RIFF", sizeof(char), 4, stream); - put_num(i_bytecount + 44 - 8, stream, 4); /* 4-7 */ - fwrite("WAVEfmt ", sizeof(char), 8, stream); /* 8-15 */ - put_num(16, stream, 4); /* 16-19 */ - put_num(1, stream, 2); /* 20-21 */ - put_num(2, stream, 2); /* 22-23 */ - put_num(44100, stream, 4); /* 24-27 */ - put_num(44100 * 2 * 2, stream, 4); /* 28-31 */ - put_num(4, stream, 2); /* 32-33 */ - put_num(16, stream, 2); /* 34-35 */ - fwrite("data", sizeof(char), 4, stream); /* 36-39 */ - put_num(i_bytecount, stream, 4); /* 40-43 */ +void RipCD::WriteWAVHeader(FILE *stream, int32_t i_bytecount) { + fwrite("RIFF", sizeof(char), 4, stream); + PutNum(i_bytecount + 44 - 8, stream, 4); /* 4-7 */ + fwrite("WAVEfmt ", sizeof(char), 8, stream); /* 8-15 */ + PutNum(16, stream, 4); /* 16-19 */ + PutNum(1, stream, 2); /* 20-21 */ + PutNum(2, stream, 2); /* 22-23 */ + PutNum(44100, stream, 4); /* 24-27 */ + PutNum(44100 * 2 * 2, stream, 4); /* 28-31 */ + PutNum(4, stream, 2); /* 32-33 */ + PutNum(16, stream, 2); /* 34-35 */ + fwrite("data", sizeof(char), 4, stream); /* 36-39 */ + PutNum(i_bytecount, stream, 4); /* 40-43 */ } -void RipCD::put_num(long int num, FILE *stream, int bytes) { - unsigned int i; - unsigned char c; +void RipCD::PutNum(int64_t num, FILE *stream, int bytes) { + unsigned int i; + unsigned char c; - for (i = 0; bytes--; i++) { - c = (num >> (i << 3)) & 0xff; - if (fwrite(&c, sizeof(char), 1, stream) == -1) { - perror("Could not write to output."); - exit(1); - } - } + for (i = 0; bytes--; i++) { + c = (num >> (i << 3)) & 0xff; + if (fwrite(&c, sizeof(char), 1, stream) == -1) { + perror("Could not write to output."); + exit(1); + } + } } -int RipCD::nTracksToRip() { - int k = 0; - for (int i = 0; i < checkboxes_.length(); i++) { - if (checkboxes_.value(i)->isChecked() == true) { - k++; - } - } - return k; +int RipCD::NumTracksToRip() { + int k = 0; + for (int i = 0; i < checkboxes_.length(); i++) { + if (checkboxes_.value(i)->isChecked() == true) { + k++; + } + } + return k; } -void RipCD::toThreadClickedRipButton() { - QString source_directory = "/tmp/"; +void RipCD::ThreadClickedRipButton() { + QString source_directory = QDir::tempPath() + "/"; - finished_success_ = 0; - finished_failed_ = 0; - ui_->progress_bar->setMaximum(nTracksToRip() * 2 * 100); + finished_success_ = 0; + finished_failed_ = 0; + ui_->progress_bar->setMaximum(NumTracksToRip() * 2 * 100); - emit(signalUpdateProgress()); + emit(SignalUpdateProgress()); - // Set up the progressbar + // Set up the progressbar - for (int i = 1; i <= i_tracks; i++) { - if (checkboxes_.value(i - 1)->isChecked() == false) { - continue; - } - tracks_to_rip_.append(i); - lsn_t i_first_lsn = cdio_get_track_lsn(p_cdio, i); - lsn_t i_last_lsn = cdio_get_track_last_lsn(p_cdio, i); -// lsn_t i_last_lsn = i_first_lsn+300; // debug + for (int i = 1; i <= i_tracks; i++) { + if (checkboxes_.value(i - 1)->isChecked() == false) { + continue; + } + tracks_to_rip_.append(i); + lsn_t i_first_lsn = cdio_get_track_lsn(p_cdio_, i); + lsn_t i_last_lsn = cdio_get_track_last_lsn(p_cdio_, i); - lsn_t i_cursor; - int16_t *p_readbuf = (int16_t *) calloc(CDIO_CD_FRAMESIZE_RAW, 1); + lsn_t i_cursor; + int16_t *p_readbuf = reinterpret_cast + (calloc(CDIO_CD_FRAMESIZE_RAW, 1)); - QString filename = source_directory + ParseFileFormatString(ui_->format_filename->text(),i) + ".wav"; - FILE *fp = fopen(filename.toUtf8().constData(), "w"); - write_WAV_header(fp, - (i_last_lsn - i_first_lsn + 1) * CDIO_CD_FRAMESIZE_RAW); - for (i_cursor = i_first_lsn; i_cursor <= i_last_lsn; i_cursor++) { - cdio_read_audio_sector(p_cdio, p_readbuf, i_cursor); - if (!p_readbuf) { - qDebug() << "Read error. Stopping."; - break; - } else { - fwrite(p_readbuf, 1, CDIO_CD_FRAMESIZE_RAW, fp); - } + QString filename = source_directory + + ParseFileFormatString(ui_->format_filename->text(), i) + ".wav"; + FILE *fp = fopen(filename.toUtf8().constData(), "w"); + WriteWAVHeader(fp, + (i_last_lsn - i_first_lsn + 1) * CDIO_CD_FRAMESIZE_RAW); - } - finished_success_++; - emit(signalUpdateProgress()); - fclose(fp); - free(p_readbuf); - p_readbuf = NULL; + for (i_cursor = i_first_lsn; i_cursor <= i_last_lsn; i_cursor++) { + cdio_read_audio_sector(p_cdio_, p_readbuf, i_cursor); + if (!p_readbuf) { + qDebug() << "Read error. Stopping."; + break; + } else { + fwrite(p_readbuf, 1, CDIO_CD_FRAMESIZE_RAW, fp); + } + } + finished_success_++; + emit(SignalUpdateProgress()); + fclose(fp); + free(p_readbuf); + p_readbuf = NULL; - //TranscoderPreset preset(Transcoder::PresetForFileType(Song::Type_OggVorbis)); - TranscoderPreset preset = - ui_->format->itemData(ui_->format->currentIndex()).value< - TranscoderPreset>(); + TranscoderPreset preset = + ui_->format->itemData(ui_->format->currentIndex()) + .value(); - QString outfilename = GetOutputFileName(filename, preset); - transcoder_->AddJob(filename.toUtf8().constData(), preset, outfilename); - } - emit (RippingComplete());} + QString outfilename = GetOutputFileName(filename, preset); + transcoder_->AddJob(filename.toUtf8().constData(), preset, outfilename); + } + emit(RippingComplete()); +} // Returns the rightmost non-empty part of 'path'. QString RipCD::TrimPath(const QString& path) const { - return path.section('/', -1, -1, QString::SectionSkipEmpty); + return path.section('/', -1, -1, QString::SectionSkipEmpty); } QString RipCD::GetOutputFileName(const QString& input, - const TranscoderPreset &preset) const { - QString path = - ui_->destination->itemData(ui_->destination->currentIndex()).toString(); - if (path.isEmpty()) { - // Keep the original path. - return input.section('.', 0, -2) + '.' + preset.extension_; - } else { - QString file_name = TrimPath(input); - file_name = file_name.section('.', 0, -2); - return path + '/' + file_name + '.' + preset.extension_; - } + const TranscoderPreset &preset) const { + QString path = + ui_->destination->itemData(ui_->destination->currentIndex()).toString(); + if (path.isEmpty()) { + // Keep the original path. + return input.section('.', 0, -2) + '.' + preset.extension_; + } else { + QString file_name = TrimPath(input); + file_name = file_name.section('.', 0, -2); + return path + '/' + file_name + '.' + preset.extension_; + } } -QString RipCD::ParseFileFormatString(const QString& file_format, int trackNo) const { - QString to_return = file_format; - to_return.replace(QString("%artist%"),ui_->artistLineEdit->text()); - to_return.replace(QString("%album%"),ui_->albumLineEdit->text()); - to_return.replace(QString("%genre%"),ui_->genreLineEdit->text()); - to_return.replace(QString("%year%"),ui_->yearLineEdit->text()); - to_return.replace(QString("%tracknum%"),QString::number(trackNo)); - to_return.replace(QString("%track%"),track_names_.value(trackNo - 1)->text()); - return to_return; +QString RipCD::ParseFileFormatString(const QString& file_format, + int trackNo) const { + QString to_return = file_format; + to_return.replace(QString("%artist%"), ui_->artistLineEdit->text()); + to_return.replace(QString("%album%"), ui_->albumLineEdit->text()); + to_return.replace(QString("%genre%"), ui_->genreLineEdit->text()); + to_return.replace(QString("%year%"), ui_->yearLineEdit->text()); + to_return.replace(QString("%tracknum%"), QString::number(trackNo)); + to_return.replace(QString("%track%"), + track_names_.value(trackNo - 1)->text()); + return to_return; } void RipCD::UpdateProgress() { - int progress = (finished_success_ + finished_failed_) * 100; - QMap current_jobs = transcoder_->GetProgress(); - foreach (float value, current_jobs.values()) { - progress += qBound(0, int(value * 100), 99); - } + int progress = (finished_success_ + finished_failed_) * 100; + QMap current_jobs = transcoder_->GetProgress(); + for (float value : current_jobs.values()) { + progress += qBound(0, static_cast(value * 100), 99); + } - ui_->progress_bar->setValue(progress); + ui_->progress_bar->setValue(progress); } -void RipCD::threadedTranscoding() { - transcoder_->Start(); - TranscoderPreset preset = - ui_->format->itemData(ui_->format->currentIndex()).value(); - // Save the last output format - QSettings s; - s.beginGroup(kSettingsGroup); - s.setValue("last_output_format", preset.extension_); - +void RipCD::ThreadedTranscoding() { + transcoder_->Start(); + TranscoderPreset preset = + ui_->format->itemData(ui_->format->currentIndex()) + .value(); + // Save the last output format + QSettings s; + s.beginGroup(kSettingsGroup); + s.setValue("last_output_format", preset.extension_); } -void RipCD::clickedRipButton() { - QtConcurrent::run(this, &RipCD::toThreadClickedRipButton); - +void RipCD::ClickedRipButton() { + QtConcurrent::run(this, &RipCD::ThreadClickedRipButton); } void RipCD::JobComplete(const QString& filename, bool success) { - (*(success ? &finished_success_ : &finished_failed_))++; - emit(signalUpdateProgress()); + (*(success ? &finished_success_ : &finished_failed_))++; + emit(SignalUpdateProgress()); } void RipCD::AllJobsComplete() { + // having a little trouble on wav files, works fine on ogg-vorbis + qSort(generated_files_); - // having a little trouble on wav files, works fine on ogg-vorbis - qSort(generated_files_); + int i; + for (i = 0; i < generated_files_.length(); i++) { + TagLib::FileRef f(generated_files_.value(i).toUtf8().constData()); - int i; - for (i = 0; i < generated_files_.length(); i++) { - TagLib::FileRef f(generated_files_.value(i).toUtf8().constData()); - - f.tag()->setTitle( - track_names_.value(tracks_to_rip_.value(i) - 1)->text().toUtf8().constData()); - f.tag()->setAlbum(ui_->albumLineEdit->text().toUtf8().constData()); - f.tag()->setArtist(ui_->artistLineEdit->text().toUtf8().constData()); - f.tag()->setGenre(ui_->genreLineEdit->text().toUtf8().constData()); - f.tag()->setYear(ui_->yearLineEdit->text().toInt()); - f.tag()->setTrack(tracks_to_rip_.value(i) - 1); - // Need to check this - //f.tag()->setDisc(ui_->discLineEdit->text().toInt()); - f.save(); - } - // Resets lists - generated_files_.clear(); - tracks_to_rip_.clear(); + f.tag()->setTitle( + track_names_.value(tracks_to_rip_.value(i) - 1) + ->text().toUtf8().constData()); + f.tag()->setAlbum(ui_->albumLineEdit->text().toUtf8().constData()); + f.tag()->setArtist(ui_->artistLineEdit->text().toUtf8().constData()); + f.tag()->setGenre(ui_->genreLineEdit->text().toUtf8().constData()); + f.tag()->setYear(ui_->yearLineEdit->text().toInt()); + f.tag()->setTrack(tracks_to_rip_.value(i) - 1); + // Need to check this + // f.tag()->setDisc(ui_->discLineEdit->text().toInt()); + f.save(); + } + // Resets lists + generated_files_.clear(); + tracks_to_rip_.clear(); } void RipCD::AppendOutput(const QString &filename) { - generated_files_.append(filename); + generated_files_.append(filename); } void RipCD::Options() { - TranscoderPreset preset = - ui_->format->itemData(ui_->format->currentIndex()).value(); + TranscoderPreset preset = + ui_->format->itemData( + ui_->format->currentIndex()) + .value(); - TranscoderOptionsDialog dialog(preset.type_, this); - if (dialog.is_valid()) { - dialog.exec(); - } + TranscoderOptionsDialog dialog(preset.type_, this); + if (dialog.is_valid()) { + dialog.exec(); + } } // Adds a folder to the destination box. void RipCD::AddDestination() { - int index = ui_->destination->currentIndex(); - QString initial_dir = ( - !ui_->destination->itemData(index).isNull() ? - ui_->destination->itemData(index).toString() : QDir::homePath()); - QString dir = QFileDialog::getExistingDirectory(this, tr("Add folder"), - initial_dir); + int index = ui_->destination->currentIndex(); + QString initial_dir = ( + !ui_->destination->itemData(index).isNull() ? + ui_->destination->itemData(index).toString() : QDir::homePath()); + QString dir = QFileDialog::getExistingDirectory(this, tr("Add folder"), + initial_dir); - if (!dir.isEmpty()) { - // Keep only a finite number of items in the box. - while (ui_->destination->count() >= kMaxDestinationItems) { - ui_->destination->removeItem(1); // The oldest folder item. - } + if (!dir.isEmpty()) { + // Keep only a finite number of items in the box. + while (ui_->destination->count() >= kMaxDestinationItems) { + ui_->destination->removeItem(1); // The oldest folder item. + } - QIcon icon = IconLoader::Load("folder"); - QVariant data = QVariant::fromValue(dir); - // Do not insert duplicates. - int duplicate_index = ui_->destination->findData(data); - if (duplicate_index == -1) { - ui_->destination->addItem(icon, dir, data); - ui_->destination->setCurrentIndex(ui_->destination->count() - 1); - } else { - ui_->destination->setCurrentIndex(duplicate_index); - } - } + QIcon icon = IconLoader::Load("folder"); + QVariant data = QVariant::fromValue(dir); + // Do not insert duplicates. + int duplicate_index = ui_->destination->findData(data); + if (duplicate_index == -1) { + ui_->destination->addItem(icon, dir, data); + ui_->destination->setCurrentIndex(ui_->destination->count() - 1); + } else { + ui_->destination->setCurrentIndex(duplicate_index); + } + } } void RipCD::Cancel() { - transcoder_->Cancel(); + transcoder_->Cancel(); } diff --git a/src/ui/ripcd.h b/src/ui/ripcd.h index 203772013..66ddd3a48 100644 --- a/src/ui/ripcd.h +++ b/src/ui/ripcd.h @@ -15,12 +15,13 @@ along with Clementine. If not, see . */ -#ifndef RIPCD_H -#define RIPCD_H +#ifndef SRC_UI_RIPCD_H_ +#define SRC_UI_RIPCD_H_ #include #include #include +#include #include #include "ui_ripcd.h" @@ -30,55 +31,51 @@ class Transcoder; struct TranscoderPreset; class RipCD: public QDialog { - Q_OBJECT -// QThread thread; + Q_OBJECT public: - RipCD(QWidget* parent = 0); - static const char* kSettingsGroup; - static const int kProgressInterval; - static const int kMaxDestinationItems; + explicit RipCD(QWidget* parent = 0); + static const char* kSettingsGroup; + static const int kProgressInterval; + static const int kMaxDestinationItems; private: - Transcoder* transcoder_; - int queued_; - int finished_success_; - int finished_failed_; - track_t i_tracks; - Ui_RipCD* ui_; - CdIo_t *p_cdio; - QList checkboxes_; - QList generated_files_; - QList tracks_to_rip_; - QList track_names_; - QString last_add_dir_; - QPushButton* cancel_button_; - - void write_WAV_header(FILE *stream, int32_t i_bytecount); - void put_num(long int num, FILE *stream, int bytes); - int nTracksToRip(); - void toThreadClickedRipButton(); - QString TrimPath(const QString& path) const; - QString GetOutputFileName(const QString& input, - const TranscoderPreset& preset) const; - QString ParseFileFormatString(const QString& file_format, int trackNo) const; - + Transcoder* transcoder_; + int queued_; + int finished_success_; + int finished_failed_; + track_t i_tracks; + Ui_RipCD* ui_; + CdIo_t *p_cdio_; + QList checkboxes_; + QList generated_files_; + QList tracks_to_rip_; + QList track_names_; + QString last_add_dir_; + QPushButton* cancel_button_; + void WriteWAVHeader(FILE *stream, int32_t i_bytecount); + void PutNum(int64_t num, FILE *stream, int bytes); + int NumTracksToRip(); + void ThreadClickedRipButton(); + QString TrimPath(const QString& path) const; + QString GetOutputFileName(const QString& input, + const TranscoderPreset& preset) const; + QString ParseFileFormatString(const QString& file_format, int trackNo) const; signals: - void RippingComplete(); - void signalUpdateProgress(); - + void RippingComplete(); + void SignalUpdateProgress(); private slots: - void UpdateProgress(); - void threadedTranscoding(); - void clickedRipButton(); - void JobComplete(const QString& filename, bool success); - void AllJobsComplete(); - void AppendOutput(const QString &filename); - void Options(); - void AddDestination(); - void Cancel(); + void UpdateProgress(); + void ThreadedTranscoding(); + void ClickedRipButton(); + void JobComplete(const QString& filename, bool success); + void AllJobsComplete(); + void AppendOutput(const QString &filename); + void Options(); + void AddDestination(); + void Cancel(); }; -#endif // RIPCD_H +#endif // SRC_UI_RIPCD_H_