/** * 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/KeyList.h" #include #include "gpg/function/GpgKeyGetter.h" #include "ui/SignalStation.h" namespace GpgFrontend::UI { KeyList::KeyList(KeyListRow::KeyType selectType, KeyListColumn::InfoType infoType, QWidget* parent) : QWidget(parent), appPath(qApp->applicationDirPath()), settings(RESOURCE_DIR(appPath) + "/conf/gpgfrontend.ini", QSettings::IniFormat), mSelectType(selectType), mInfoType(infoType) { mKeyList = new QTableWidget(this); mKeyList->setColumnCount(7); mKeyList->horizontalHeader()->setSectionResizeMode( QHeaderView::ResizeToContents); mKeyList->verticalHeader()->hide(); mKeyList->setShowGrid(false); mKeyList->sortByColumn(2, Qt::AscendingOrder); mKeyList->setSelectionBehavior(QAbstractItemView::SelectRows); mKeyList->setSelectionMode(QAbstractItemView::SingleSelection); // table items not editable mKeyList->setEditTriggers(QAbstractItemView::NoEditTriggers); // no focus (rectangle around table items) // maybe it should focus on whole row mKeyList->setFocusPolicy(Qt::NoFocus); mKeyList->setAlternatingRowColors(true); // Hidden Column For Purpose if (!(mInfoType & KeyListColumn::TYPE)) { mKeyList->setColumnHidden(1, true); } if (!(mInfoType & KeyListColumn::NAME)) { mKeyList->setColumnHidden(2, true); } if (!(mInfoType & KeyListColumn::EmailAddress)) { mKeyList->setColumnHidden(3, true); } if (!(mInfoType & KeyListColumn::Usage)) { mKeyList->setColumnHidden(4, true); } if (!(mInfoType & KeyListColumn::Validity)) { mKeyList->setColumnHidden(5, true); } if (!(mInfoType & KeyListColumn::FingerPrint)) { mKeyList->setColumnHidden(6, true); } QStringList labels; labels << _("Select") << _("Type") << _("Name") << _("Email Address") << _("Usage") << _("Validity") << _("Finger Print"); mKeyList->setHorizontalHeaderLabels(labels); mKeyList->horizontalHeader()->setStretchLastSection(false); auto* layout = new QVBoxLayout; layout->addWidget(mKeyList); layout->setContentsMargins(0, 0, 0, 0); layout->setSpacing(3); setLayout(layout); popupMenu = new QMenu(this); // register key database refresh signal connect(SignalStation::GetInstance(), SIGNAL(KeyDatabaseRefresh()), this, SLOT(slotRefresh())); connect(mKeyList, SIGNAL(doubleClicked(const QModelIndex&)), this, SLOT(slotDoubleClicked(const QModelIndex&))); setAcceptDrops(true); slotRefresh(); } void KeyList::slotRefresh() { LOG(INFO) << "KeyList::slotRefresh Called"; auto keyList = getChecked(); // while filling the table, sort enabled causes errors mKeyList->setSortingEnabled(false); mKeyList->clearContents(); auto keys = GpgKeyGetter::GetInstance().FetchKey(); auto it = keys->begin(); int row_count = 0; while (it != keys->end()) { if (mFilter != nullptr) { if (!mFilter(*it)) { it = keys->erase(it); continue; } } if (!excluded_key_ids.empty()) { auto iterator = std::find_if(excluded_key_ids.begin(), excluded_key_ids.end(), [it](const auto& key_id) -> bool { if (it->id() == key_id) return true; else return false; }); if (iterator != excluded_key_ids.end()) { it = keys->erase(it); continue; } } if (mSelectType == KeyListRow::ONLY_SECRET_KEY && !it->is_private_key()) { it = keys->erase(it); continue; } row_count++; it++; } mKeyList->setRowCount(row_count); int row_index = 0; it = keys->begin(); buffered_keys.clear(); while (it != keys->end()) { buffered_keys.push_back(GpgKeyGetter::GetInstance().GetKey(it->id())); auto* tmp0 = new QTableWidgetItem(QString::number(row_index)); tmp0->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable); tmp0->setTextAlignment(Qt::AlignCenter); tmp0->setCheckState(Qt::Unchecked); mKeyList->setItem(row_index, 0, tmp0); QString type_str; QTextStream type_steam(&type_str); if (it->is_private_key()) { type_steam << "pub/sec"; } else { type_steam << "pub"; } if (it->is_private_key() && !it->has_master_key()) { type_steam << "#"; } auto* tmp1 = new QTableWidgetItem(type_str); mKeyList->setItem(row_index, 1, tmp1); auto* tmp2 = new QTableWidgetItem(QString::fromStdString(it->name())); mKeyList->setItem(row_index, 2, tmp2); auto* tmp3 = new QTableWidgetItem(QString::fromStdString(it->email())); mKeyList->setItem(row_index, 3, tmp3); QString usage; QTextStream usage_steam(&usage); if (it->CanCertActual()) usage_steam << "C"; if (it->CanEncrActual()) usage_steam << "E"; if (it->CanSignActual()) usage_steam << "S"; if (it->CanAuthActual()) usage_steam << "A"; auto* temp_usage = new QTableWidgetItem(usage); temp_usage->setTextAlignment(Qt::AlignCenter); mKeyList->setItem(row_index, 4, temp_usage); auto* temp_validity = new QTableWidgetItem(QString::fromStdString(it->owner_trust())); temp_validity->setTextAlignment(Qt::AlignCenter); mKeyList->setItem(row_index, 5, temp_validity); auto* temp_fpr = new QTableWidgetItem(QString::fromStdString(it->fpr())); temp_fpr->setTextAlignment(Qt::AlignCenter); mKeyList->setItem(row_index, 6, temp_fpr); // strike out expired keys if (it->expired() || it->revoked()) { QFont strike = tmp2->font(); strike.setStrikeOut(true); tmp0->setFont(strike); temp_usage->setFont(strike); temp_fpr->setFont(strike); temp_validity->setFont(strike); tmp1->setFont(strike); tmp2->setFont(strike); tmp3->setFont(strike); } it++; ++row_index; } setChecked(keyList); } KeyIdArgsListPtr KeyList::getChecked() { auto ret = std::make_unique(); for (int i = 0; i < mKeyList->rowCount(); i++) { if (mKeyList->item(i, 0)->checkState() == Qt::Checked) { ret->push_back(buffered_keys[i].id()); } } return ret; } KeyIdArgsListPtr KeyList::getAllPrivateKeys() { auto ret = std::make_unique(); for (int i = 0; i < mKeyList->rowCount(); i++) { if (mKeyList->item(i, 1) && buffered_keys[i].is_private_key()) { ret->push_back(buffered_keys[i].id()); } } return ret; } KeyIdArgsListPtr KeyList::getPrivateChecked() { auto ret = std::make_unique(); for (int i = 0; i < mKeyList->rowCount(); i++) { if ((mKeyList->item(i, 0)->checkState() == Qt::Checked) && (mKeyList->item(i, 1))) { ret->push_back(buffered_keys[i].id()); } } return ret; } void KeyList::setChecked(const KeyIdArgsListPtr& keyIds) { if (!keyIds->empty()) { for (int i = 0; i < mKeyList->rowCount(); i++) { if (std::find(keyIds->begin(), keyIds->end(), buffered_keys[i].id()) != keyIds->end()) { mKeyList->item(i, 0)->setCheckState(Qt::Checked); } } } } KeyIdArgsListPtr KeyList::getSelected() { auto ret = std::make_unique(); for (int i = 0; i < mKeyList->rowCount(); i++) { if (mKeyList->item(i, 0)->isSelected() == 1) { ret->push_back(buffered_keys[i].id()); } } return ret; } [[maybe_unused]] bool KeyList::containsPrivateKeys() { for (int i = 0; i < mKeyList->rowCount(); i++) { if (mKeyList->item(i, 1)) { return true; } } return false; } void KeyList::setColumnWidth(int row, int size) { mKeyList->setColumnWidth(row, size); } void KeyList::contextMenuEvent(QContextMenuEvent* event) { if (mKeyList->selectedItems().length() > 0) { popupMenu->exec(event->globalPos()); } } void KeyList::addSeparator() { popupMenu->addSeparator(); } void KeyList::addMenuAction(QAction* act) { popupMenu->addAction(act); } void KeyList::dropEvent(QDropEvent* event) { auto* dialog = new QDialog(); dialog->setWindowTitle(_("Import Keys")); QLabel* label; label = new QLabel(QString(_("You've dropped something on the table.")) + "\n " + _("GpgFrontend " "will now try to import key(s).") + "\n"); // "always import keys"-CheckBox auto* checkBox = new QCheckBox(_("Always import without bothering.")); if (settings.value("general/confirmImportKeys").toBool()) checkBox->setCheckState(Qt::Unchecked); // Buttons for ok and cancel auto* buttonBox = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel); connect(buttonBox, SIGNAL(accepted()), dialog, SLOT(accept())); connect(buttonBox, SIGNAL(rejected()), dialog, SLOT(reject())); auto* vbox = new QVBoxLayout(); vbox->addWidget(label); vbox->addWidget(checkBox); vbox->addWidget(buttonBox); dialog->setLayout(vbox); if (settings.value("general/confirmImportKeys", Qt::Checked).toBool()) { dialog->exec(); if (dialog->result() == QDialog::Rejected) { return; } if (checkBox->isChecked()) { settings.setValue("general/confirmImportKeys", false); } else { settings.setValue("general/confirmImportKeys", true); } } if (event->mimeData()->hasUrls()) { for (const QUrl& tmp : event->mimeData()->urls()) { QFile file; file.setFileName(tmp.toLocalFile()); if (!file.open(QIODevice::ReadOnly)) { LOG(INFO) << _("Couldn't Open File") << ":" << tmp.toString().toStdString(); } QByteArray inBuffer = file.readAll(); this->importKeys(inBuffer); file.close(); } } else { QByteArray inBuffer(event->mimeData()->text().toUtf8()); this->importKeys(inBuffer); } } void KeyList::dragEnterEvent(QDragEnterEvent* event) { event->acceptProposedAction(); } /** set background color for Keys and put them to top * */ [[maybe_unused]] void KeyList::markKeys(QStringList* keyIds) { foreach (QString id, *keyIds) { qDebug() << "marked: " << id; } } void KeyList::importKeys(const QByteArray& inBuffer) { auto std_buffer = std::make_unique(inBuffer.toStdString()); GpgImportInformation result = GpgKeyImportExportor::GetInstance().ImportKey(std::move(std_buffer)); new KeyImportDetailDialog(result, false, this); } void KeyList::setExcludeKeys(std::initializer_list key_ids) { excluded_key_ids.clear(); for (auto& key_id : key_ids) { excluded_key_ids.push_back(key_id); } } void KeyList::setFilter(std::function filter) { this->mFilter = std::move(filter); } void KeyList::slotDoubleClicked(const QModelIndex& index) { if (mAction != nullptr) { const auto key = GpgKeyGetter::GetInstance().GetKey(buffered_keys[index.row()].id()); mAction(key, this); } } void KeyList::setDoubleClickedAction( std::function action) { this->mAction = std::move(action); } std::string KeyList::getSelectedKey() { for (int i = 0; i < mKeyList->rowCount(); i++) { if (mKeyList->item(i, 0)->isSelected() == 1) { return buffered_keys[i].id(); } } return {}; } } // namespace GpgFrontend::UI