/** * This file is part of GpgFrontend. * * GpgFrontend 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. * * Foobar 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 Foobar. If not, see . * * The initial version of the source code is inherited from gpg4usb-team. * Their source code version also complies with GNU General Public License. * * The source code version of this software was modified and released * by Saturneric starting on May 12, 2021. * */ #include "ui/widgets/FilePage.h" #include #include #include "ui/MainWindow.h" #include "ui/SignalStation.h" #include "ui_FilePage.h" namespace GpgFrontend::UI { FilePage::FilePage(QWidget* parent) : QWidget(parent), ui(std::make_shared()) { ui->setupUi(this); firstParent = parent; dirModel = new QFileSystemModel(); dirModel->setRootPath(QDir::currentPath()); dirModel->setFilter(QDir::AllDirs | QDir::Files | QDir::NoDotAndDotDot); ui->fileTreeView->setModel(dirModel); ui->fileTreeView->setColumnWidth(0, 320); ui->fileTreeView->sortByColumn(0, Qt::AscendingOrder); mPath = boost::filesystem::path(dirModel->rootPath().toStdString()); createPopupMenu(); connect(ui->upPathButton, &QPushButton::clicked, this, &FilePage::slotUpLevel); connect(ui->refreshButton, &QPushButton::clicked, this, &FilePage::slotGoPath); ui->optionsButton->setMenu(optionPopUpMenu); ui->pathEdit->setText(dirModel->rootPath()); pathEditCompleter = new QCompleter(this); pathCompleteModel = new QStringListModel(); pathEditCompleter->setModel(pathCompleteModel); pathEditCompleter->setCaseSensitivity(Qt::CaseInsensitive); pathEditCompleter->setCompletionMode(QCompleter::UnfilteredPopupCompletion); ui->pathEdit->setCompleter(pathEditCompleter); connect(ui->fileTreeView, &QTreeView::clicked, this, &FilePage::fileTreeViewItemClicked); connect(ui->fileTreeView, &QTreeView::doubleClicked, this, &FilePage::fileTreeViewItemDoubleClicked); connect(ui->fileTreeView, &QTreeView::customContextMenuRequested, this, &FilePage::onCustomContextMenu); connect(ui->pathEdit, &QLineEdit::textChanged, [=]() { auto path = ui->pathEdit->text(); auto dir = QDir(path); if (path.endsWith("/") && dir.isReadable()) { auto dir_list = dir.entryInfoList(QDir::AllEntries); QStringList paths; for (int i = 1; i < dir_list.size(); i++) { const auto file_path = dir_list.at(i).filePath(); const auto file_name = dir_list.at(i).fileName(); if (file_name == "." || file_name == "..") continue; paths.append(file_path); } pathCompleteModel->setStringList(paths); } }); connect(this, &FilePage::signalRefreshInfoBoard, SignalStation::GetInstance(), &SignalStation::signalRefreshInfoBoard); } void FilePage::fileTreeViewItemClicked(const QModelIndex& index) { selectedPath = boost::filesystem::path( dirModel->fileInfo(index).absoluteFilePath().toStdString()); mPath = selectedPath; LOG(INFO) << "selected path" << selectedPath; } void FilePage::slotUpLevel() { QModelIndex currentRoot = ui->fileTreeView->rootIndex(); auto utf8_path = dirModel->fileInfo(currentRoot).absoluteFilePath().toStdString(); boost::filesystem::path path_obj(utf8_path); mPath = path_obj; LOG(INFO) << "get path" << mPath; if (mPath.has_parent_path() && !mPath.parent_path().empty()) { mPath = mPath.parent_path(); LOG(INFO) << "parent path" << mPath; ui->pathEdit->setText(mPath.string().c_str()); this->slotGoPath(); } } void FilePage::fileTreeViewItemDoubleClicked(const QModelIndex& index) { QFileInfo file_info(dirModel->fileInfo(index).absoluteFilePath()); if (file_info.isFile()) { slotOpenItem(); } else { ui->pathEdit->setText(file_info.filePath()); slotGoPath(); } } QString FilePage::getSelected() const { return QString::fromStdString(selectedPath.string()); } void FilePage::slotGoPath() { const auto path_edit = ui->pathEdit->text().toStdString(); boost::filesystem::path path_obj(path_edit); if (mPath.string() != path_edit) mPath = path_obj; auto fileInfo = QFileInfo(mPath.string().c_str()); if (fileInfo.isDir() && fileInfo.isReadable() && fileInfo.isExecutable()) { mPath = boost::filesystem::path(fileInfo.filePath().toStdString()); LOG(INFO) << "set path" << mPath; ui->fileTreeView->setRootIndex(dirModel->index(fileInfo.filePath())); dirModel->setRootPath(fileInfo.filePath()); for (int i = 1; i < dirModel->columnCount(); ++i) { ui->fileTreeView->resizeColumnToContents(i); } ui->pathEdit->setText(mPath.generic_path().string().c_str()); } else { QMessageBox::critical( this, _("Error"), _("The path is not exists, unprivileged or unreachable.")); } emit pathChanged(mPath.string().c_str()); } void FilePage::createPopupMenu() { popUpMenu = new QMenu(); openItemAct = new QAction(_("Open"), this); connect(openItemAct, &QAction::triggered, this, &FilePage::slotOpenItem); renameItemAct = new QAction(_("Rename"), this); connect(renameItemAct, &QAction::triggered, this, &FilePage::slotRenameItem); deleteItemAct = new QAction(_("Delete"), this); connect(deleteItemAct, &QAction::triggered, this, &FilePage::slotDeleteItem); encryptItemAct = new QAction(_("Encrypt Sign"), this); connect(encryptItemAct, &QAction::triggered, this, &FilePage::slotEncryptItem); decryptItemAct = new QAction(QString(_("Decrypt Verify")) + " " + _("(.gpg .asc)"), this); connect(decryptItemAct, &QAction::triggered, this, &FilePage::slotDecryptItem); signItemAct = new QAction(_("Sign"), this); connect(signItemAct, &QAction::triggered, this, &FilePage::slotSignItem); verifyItemAct = new QAction(QString(_("Verify")) + " " + _("(.sig .gpg .asc)"), this); connect(verifyItemAct, &QAction::triggered, this, &FilePage::slotVerifyItem); hashCalculateAct = new QAction(_("Calculate Hash"), this); connect(hashCalculateAct, &QAction::triggered, this, &FilePage::slotCalculateHash); mkdirAct = new QAction(_("Make New Directory"), this); connect(mkdirAct, &QAction::triggered, this, &FilePage::slotMkdir); createEmptyFileAct = new QAction(_("Create Empty File"), this); connect(createEmptyFileAct, &QAction::triggered, this, &FilePage::slotCreateEmptyFile); popUpMenu->addAction(openItemAct); popUpMenu->addAction(renameItemAct); popUpMenu->addAction(deleteItemAct); popUpMenu->addSeparator(); popUpMenu->addAction(encryptItemAct); popUpMenu->addAction(decryptItemAct); popUpMenu->addAction(signItemAct); popUpMenu->addAction(verifyItemAct); popUpMenu->addSeparator(); popUpMenu->addAction(mkdirAct); popUpMenu->addAction(createEmptyFileAct); popUpMenu->addAction(hashCalculateAct); optionPopUpMenu = new QMenu(); auto showHiddenAct = new QAction(_("Show Hidden File"), this); showHiddenAct->setCheckable(true); connect(showHiddenAct, &QAction::triggered, this, [&](bool checked) { LOG(INFO) << "Set Hidden" << checked; if (checked) dirModel->setFilter(dirModel->filter() | QDir::Hidden); else dirModel->setFilter(dirModel->filter() & ~QDir::Hidden); dirModel->setRootPath(mPath.string().c_str()); }); optionPopUpMenu->addAction(showHiddenAct); auto showSystemAct = new QAction(_("Show System File"), this); showSystemAct->setCheckable(true); connect(showSystemAct, &QAction::triggered, this, [&](bool checked) { LOG(INFO) << "Set Hidden" << checked; if (checked) dirModel->setFilter(dirModel->filter() | QDir::System); else dirModel->setFilter(dirModel->filter() & ~QDir::System); dirModel->setRootPath(mPath.string().c_str()); }); optionPopUpMenu->addAction(showSystemAct); } void FilePage::onCustomContextMenu(const QPoint& point) { QModelIndex index = ui->fileTreeView->indexAt(point); selectedPath = boost::filesystem::path( dirModel->fileInfo(index).absoluteFilePath().toStdString()); LOG(INFO) << "right click" << selectedPath; if (index.isValid()) { openItemAct->setEnabled(true); renameItemAct->setEnabled(true); deleteItemAct->setEnabled(true); QFileInfo info(QString::fromStdString(selectedPath.string())); encryptItemAct->setEnabled( info.isFile() && (info.suffix() != "gpg" && info.suffix() != "sig")); encryptItemAct->setEnabled( info.isFile() && (info.suffix() != "gpg" && info.suffix() != "sig")); decryptItemAct->setEnabled(info.isFile() && info.suffix() == "gpg"); signItemAct->setEnabled(info.isFile() && (info.suffix() != "gpg" && info.suffix() != "sig")); verifyItemAct->setEnabled( info.isFile() && (info.suffix() == "sig" || info.suffix() == "gpg")); hashCalculateAct->setEnabled(info.isFile() && info.isReadable()); } else { openItemAct->setEnabled(false); renameItemAct->setEnabled(false); deleteItemAct->setEnabled(false); encryptItemAct->setEnabled(false); encryptItemAct->setEnabled(false); decryptItemAct->setEnabled(false); signItemAct->setEnabled(false); verifyItemAct->setEnabled(false); hashCalculateAct->setEnabled(false); } popUpMenu->exec(ui->fileTreeView->viewport()->mapToGlobal(point)); } void FilePage::slotOpenItem() { QFileInfo info(QString::fromStdString(selectedPath.string())); if (info.isDir()) { if (info.isReadable() && info.isExecutable()) { const auto file_path = info.filePath().toStdString(); LOG(INFO) << "set path" << file_path; ui->pathEdit->setText(info.filePath()); slotGoPath(); } else { QMessageBox::critical(this, _("Error"), _("The directory is unprivileged or unreachable.")); } } else { if (info.isReadable()) { auto mainWindow = qobject_cast(firstParent); LOG(INFO) << "open item" << selectedPath; auto qt_path = QString::fromStdString(selectedPath.string()); if (mainWindow != nullptr) mainWindow->slotOpenFile(qt_path); } else { QMessageBox::critical(this, _("Error"), _("The file is unprivileged or unreachable.")); } } } void FilePage::slotRenameItem() { auto new_name_path = selectedPath, old_name_path = selectedPath; auto old_name = old_name_path.filename(); new_name_path = new_name_path.remove_filename(); bool ok; auto text = QInputDialog::getText(this, _("Rename"), _("New Filename"), QLineEdit::Normal, old_name.string().c_str(), &ok); if (ok && !text.isEmpty()) { try { new_name_path /= text.toStdString(); LOG(INFO) << "new name path" << new_name_path; boost::filesystem::rename(old_name_path, new_name_path); // refresh this->slotGoPath(); } catch (...) { LOG(ERROR) << "rename error" << new_name_path; QMessageBox::critical(this, _("Error"), _("Unable to rename the file or folder.")); } } } void FilePage::slotDeleteItem() { QModelIndex index = ui->fileTreeView->currentIndex(); QVariant data = ui->fileTreeView->model()->data(index); auto ret = QMessageBox::warning(this, _("Warning"), _("Are you sure you want to delete it?"), QMessageBox::Ok | QMessageBox::Cancel); if (ret == QMessageBox::Cancel) return; LOG(INFO) << "Delete Item" << data.toString().toStdString(); if (!dirModel->remove(index)) { QMessageBox::critical(this, _("Error"), _("Unable to delete the file or folder.")); } } void FilePage::slotEncryptItem() { auto mainWindow = qobject_cast(firstParent); if (mainWindow != nullptr) mainWindow->slotFileEncryptSign(); } void FilePage::slotDecryptItem() { auto mainWindow = qobject_cast(firstParent); if (mainWindow != nullptr) mainWindow->slotFileDecryptVerify(); } void FilePage::slotSignItem() { auto mainWindow = qobject_cast(firstParent); if (mainWindow != nullptr) mainWindow->slotFileSign(); } void FilePage::slotVerifyItem() { auto mainWindow = qobject_cast(firstParent); if (mainWindow != nullptr) mainWindow->slotFileVerify(); } void FilePage::slotCalculateHash() { // Returns empty QByteArray() on failure. QFileInfo info(QString::fromStdString(selectedPath.string())); if (info.isFile() && info.isReadable()) { std::stringstream ss; ss << "[#] " << _("File Hash Information") << std::endl; ss << " " << _("filename") << _(": ") << selectedPath.filename().string().c_str() << std::endl; QFile f(info.filePath()); f.open(QFile::ReadOnly); auto buffer = f.readAll(); LOG(INFO) << "buffer size" << buffer.size(); f.close(); if (f.open(QFile::ReadOnly)) { auto hash_md5 = QCryptographicHash(QCryptographicHash::Md5); // md5 hash_md5.addData(buffer); auto md5 = hash_md5.result().toHex().toStdString(); LOG(INFO) << "md5" << md5; ss << " " << "md5" << _(": ") << md5 << std::endl; auto hash_sha1 = QCryptographicHash(QCryptographicHash::Sha1); // sha1 hash_sha1.addData(buffer); auto sha1 = hash_sha1.result().toHex().toStdString(); LOG(INFO) << "sha1" << sha1; ss << " " << "sha1" << _(": ") << sha1 << std::endl; auto hash_sha256 = QCryptographicHash(QCryptographicHash::Sha256); // sha1 hash_sha256.addData(buffer); auto sha256 = hash_sha256.result().toHex().toStdString(); LOG(INFO) << "sha256" << sha256; ss << " " << "sha256" << _(": ") << sha256 << std::endl; ss << std::endl; emit signalRefreshInfoBoard(ss.str().c_str(), InfoBoardStatus::INFO_ERROR_OK); } } } void FilePage::slotMkdir() { auto index = ui->fileTreeView->rootIndex(); QString new_dir_name; bool ok; new_dir_name = QInputDialog::getText(this, _("Make New Directory"), _("Directory Name"), QLineEdit::Normal, new_dir_name, &ok); if (ok && !new_dir_name.isEmpty()) { dirModel->mkdir(index, new_dir_name); } } void FilePage::slotCreateEmptyFile() { auto root_path_str = dirModel->rootPath().toStdString(); boost::filesystem::path root_path(root_path_str); QString new_file_name; bool ok; new_file_name = QInputDialog::getText(this, _("Create Empty File"), _("Filename (you can given extension)"), QLineEdit::Normal, new_file_name, &ok); if (ok && !new_file_name.isEmpty()) { auto file_path = root_path / new_file_name.toStdString(); QFile new_file(file_path.string().c_str()); if (!new_file.open(QIODevice::WriteOnly | QIODevice::NewOnly)) { QMessageBox::critical(this, _("Error"), _("Unable to create the file.")); } new_file.close(); } } void FilePage::keyPressEvent(QKeyEvent* event) { LOG(INFO) << "Key Press" << event->key(); if (ui->pathEdit->hasFocus() && (event->key() == Qt::Key_Return || event->key() == Qt::Key_Enter)) { slotGoPath(); } else if (ui->fileTreeView->currentIndex().isValid()) { if (event->key() == Qt::Key_Return || event->key() == Qt::Key_Enter) slotOpenItem(); else if (event->key() == Qt::Key_Delete || event->key() == Qt::Key_Backspace) slotDeleteItem(); } } } // namespace GpgFrontend::UI