/** * 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/KeyList.h" #include KeyList::KeyList(GpgME::GpgContext *ctx, QWidget *parent) : QWidget(parent) { mCtx = ctx; 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); // tableitems not editable mKeyList->setEditTriggers(QAbstractItemView::NoEditTriggers); // no focus (rectangle around tableitems) // may be it should focus on whole row mKeyList->setFocusPolicy(Qt::NoFocus); mKeyList->setAlternatingRowColors(true); QStringList labels; labels << tr("Select") << tr("Type") << tr("Name") << tr("Email Address") << tr("Usage") << tr("Validity") << tr("Finger Print"); mKeyList->setHorizontalHeaderLabels(labels); mKeyList->horizontalHeader()->setStretchLastSection(true); auto *layout = new QVBoxLayout; layout->addWidget(mKeyList); layout->setContentsMargins(0, 0, 0, 0); layout->setSpacing(3); setLayout(layout); popupMenu = new QMenu(this); connect(mCtx, SIGNAL(signalKeyDBChanged()), this, SLOT(slotRefresh())); setAcceptDrops(true); slotRefresh(); } void KeyList::slotRefresh() { QStringList *keyList; keyList = getChecked(); // while filling the table, sort enabled causes errors mKeyList->setSortingEnabled(false); mKeyList->clearContents(); GpgKeyList keys = mCtx->listKeys(); mKeyList->setRowCount(keys.size()); int row = 0; GpgKeyList::iterator it = keys.begin(); buffered_keys.clear(); while (it != keys.end()) { buffered_keys.push_back(*it); auto *tmp0 = new QTableWidgetItem(QString::number(row)); tmp0->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable); tmp0->setTextAlignment(Qt::AlignCenter); tmp0->setCheckState(Qt::Unchecked); mKeyList->setItem(row, 0, tmp0); if (it->is_private_key) { auto *tmp1 = new QTableWidgetItem("pub/sec"); mKeyList->setItem(row, 1, tmp1); } else { auto *tmp1 = new QTableWidgetItem("pub"); mKeyList->setItem(row, 1, tmp1); } auto *tmp2 = new QTableWidgetItem(it->name); tmp2->setToolTip(it->name); mKeyList->setItem(row, 2, tmp2); auto *tmp3 = new QTableWidgetItem(it->email); tmp3->setToolTip(it->email); // strike out expired keys if(it->expired || it->revoked) { QFont strike = tmp2->font(); strike.setStrikeOut(true); tmp2->setFont(strike); tmp3->setFont(strike); } mKeyList->setItem(row, 3, tmp3); QString usage; QTextStream usage_steam(&usage); if(it->can_certify) usage_steam << "C"; if(it->can_encrypt) usage_steam << "E"; if(it->can_sign) usage_steam << "S"; if(it->can_authenticate) usage_steam << "A"; auto *temp_usage = new QTableWidgetItem(usage); temp_usage->setTextAlignment(Qt::AlignCenter); mKeyList->setItem(row, 4, temp_usage); auto *temp_validity = new QTableWidgetItem(it->owner_trust); temp_validity->setTextAlignment(Qt::AlignCenter); mKeyList->setItem(row, 5, temp_validity); auto *temp_fpr = new QTableWidgetItem(it->fpr); temp_fpr->setTextAlignment(Qt::AlignCenter); mKeyList->setItem(row, 6, temp_fpr); it++; ++row; } mKeyList->setSortingEnabled(true); setChecked(keyList); } QStringList *KeyList::getChecked() { auto *ret = new QStringList(); for (int i = 0; i < mKeyList->rowCount(); i++) { if (mKeyList->item(i, 0)->checkState() == Qt::Checked) { *ret << buffered_keys[i].id; } } return ret; } QStringList *KeyList::getAllPrivateKeys() { auto *ret = new QStringList(); for (int i = 0; i < mKeyList->rowCount(); i++) { if (mKeyList->item(i, 1)) { *ret << buffered_keys[i].id; } } return ret; } QStringList *KeyList::getPrivateChecked() { auto *ret = new QStringList(); for (int i = 0; i < mKeyList->rowCount(); i++) { if ((mKeyList->item(i, 0)->checkState() == Qt::Checked) && (mKeyList->item(i, 1))) { *ret << buffered_keys[i].id; } } return ret; } void KeyList::setChecked(QStringList *keyIds) { if (!keyIds->isEmpty()) { for (int i = 0; i < mKeyList->rowCount(); i++) { if (keyIds->contains(buffered_keys[i].id)) { mKeyList->item(i, 0)->setCheckState(Qt::Checked); } } } } QStringList *KeyList::getSelected() { auto *ret = new QStringList(); for (int i = 0; i < mKeyList->rowCount(); i++) { if (mKeyList->item(i, 0)->isSelected() == 1) { *ret << 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) { popupMenu->exec(event->globalPos()); } void KeyList::addMenuAction(QAction *act) { popupMenu->addAction(act); } void KeyList::dropEvent(QDropEvent* event) { // importKeyDialog(); QSettings settings; auto *dialog = new QDialog(); dialog->setWindowTitle(tr("Import Keys")); QLabel *label; label = new QLabel(tr("You've dropped something on the keylist.\n gpg4usb will now try to import key(s).")+"\n"); // "always import keys"-CheckBox auto *checkBox = new QCheckBox(tr("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()) { foreach (QUrl tmp, event->mimeData()->urls()) { QFile file; file.setFileName(tmp.toLocalFile()); if (!file.open(QIODevice::ReadOnly)) { qDebug() << tr("Couldn't Open File: ") + tmp.toString(); } 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(QByteArray inBuffer) { GpgImportInformation result = mCtx->importKey(std::move(inBuffer)); new KeyImportDetailDialog(mCtx, result, this); } void KeyList::uploadKeyToServer(QByteArray *keys) { QUrl reqUrl("http://localhost:11371/pks/add"); qnam = new QNetworkAccessManager(this); QUrl params; keys->replace("\n", "%0D%0A") .replace("(", "%28") .replace(")", "%29") .replace("/", "%2F") .replace(":", "%3A") .replace("+","%2B") .replace(' ', '+'); QUrlQuery q; q.addQueryItem("keytext", *keys); params = q.query(QUrl::FullyEncoded).toUtf8(); QNetworkRequest req(reqUrl); req.setHeader(QNetworkRequest::ContentTypeHeader,"application/x-www-form-urlencoded"); QNetworkReply *reply = qnam->post(req,params.toEncoded()); connect(reply, SIGNAL(finished()), this, SLOT(uploadFinished())); qDebug() << "REQURL: " << reqUrl; qDebug() << "PARAMS.ENCODED: " << params.toEncoded(); } void KeyList::uploadFinished() { auto *reply = qobject_cast(sender()); QByteArray response = reply->readAll(); qDebug() << "RESPNOSE: " << response.data(); //reply->readAll(); qDebug() << "ERROR: " << reply->error(); if (reply->error()) { qDebug() << "Error while contacting keyserver!"; return; } else { qDebug() << "Success while contacting keyserver!"; } reply->deleteLater(); }