File indexing completed on 2024-12-08 03:40:51
0001 /* 0002 This file is part of the KDE libraries 0003 SPDX-FileCopyrightText: 2020 Ahmad Samir <a.samirh78@gmail.com> 0004 0005 SPDX-License-Identifier: LGPL-2.0-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL 0006 */ 0007 0008 #include "widgetsaskuseractionhandler.h" 0009 0010 #include <KConfig> 0011 #include <KConfigGroup> 0012 #include <KGuiItem> 0013 #include <KIO/WorkerBase> 0014 #include <KJob> 0015 #include <KJobWidgets> 0016 #include <KLocalizedString> 0017 #include <KMessageDialog> 0018 #include <KSharedConfig> 0019 #include <KSslInfoDialog> 0020 #include <KStandardGuiItem> 0021 #include <kio_widgets_debug.h> 0022 0023 #include <QApplication> 0024 #include <QDialogButtonBox> 0025 #include <QPointer> 0026 #include <QRegularExpression> 0027 #include <QUrl> 0028 0029 class KIO::WidgetsAskUserActionHandlerPrivate 0030 { 0031 public: 0032 explicit WidgetsAskUserActionHandlerPrivate(WidgetsAskUserActionHandler *qq) 0033 : q(qq) 0034 { 0035 } 0036 0037 // Creates a KSslInfoDialog or falls back to a generic Information dialog 0038 void sslMessageBox(const QString &text, const KIO::MetaData &metaData, QWidget *parent); 0039 0040 bool gotPersistentUserReply(KIO::AskUserActionInterface::MessageDialogType type, const KConfigGroup &cg, const QString &dontAskAgainName); 0041 void savePersistentUserReply(KIO::AskUserActionInterface::MessageDialogType type, KConfigGroup &cg, const QString &dontAskAgainName, int result); 0042 0043 WidgetsAskUserActionHandler *const q; 0044 QPointer<QWidget> m_parentWidget = nullptr; 0045 0046 QWidget *getParentWidget(KJob *job); 0047 QWidget *getParentWidget(QWidget *widget); 0048 }; 0049 0050 bool KIO::WidgetsAskUserActionHandlerPrivate::gotPersistentUserReply(KIO::AskUserActionInterface::MessageDialogType type, 0051 const KConfigGroup &cg, 0052 const QString &dontAskAgainName) 0053 { 0054 // storage values matching the logic of FrameworkIntegration's KMessageBoxDontAskAgainConfigStorage 0055 switch (type) { 0056 case KIO::AskUserActionInterface::QuestionTwoActions: 0057 case KIO::AskUserActionInterface::QuestionTwoActionsCancel: 0058 case KIO::AskUserActionInterface::WarningTwoActions: 0059 case KIO::AskUserActionInterface::WarningTwoActionsCancel: { 0060 // storage holds "true" if persistent reply is "Yes", "false" for persistent "No", 0061 // otherwise no persistent reply is present 0062 const QString value = cg.readEntry(dontAskAgainName, QString()); 0063 if ((value.compare(QLatin1String("yes"), Qt::CaseInsensitive) == 0) || (value.compare(QLatin1String("true"), Qt::CaseInsensitive) == 0)) { 0064 Q_EMIT q->messageBoxResult(KIO::WorkerBase::PrimaryAction); 0065 return true; 0066 } 0067 if ((value.compare(QLatin1String("no"), Qt::CaseInsensitive) == 0) || (value.compare(QLatin1String("false"), Qt::CaseInsensitive) == 0)) { 0068 Q_EMIT q->messageBoxResult(KIO::WorkerBase::SecondaryAction); 0069 return true; 0070 } 0071 break; 0072 } 0073 case KIO::AskUserActionInterface::WarningContinueCancel: { 0074 // storage holds "false" if persistent reply is "Continue" 0075 // otherwise no persistent reply is present 0076 const bool value = cg.readEntry(dontAskAgainName, true); 0077 if (value == false) { 0078 Q_EMIT q->messageBoxResult(KIO::WorkerBase::Continue); 0079 return true; 0080 } 0081 break; 0082 } 0083 default: 0084 break; 0085 } 0086 0087 return false; 0088 } 0089 0090 void KIO::WidgetsAskUserActionHandlerPrivate::savePersistentUserReply(KIO::AskUserActionInterface::MessageDialogType type, 0091 KConfigGroup &cg, 0092 const QString &dontAskAgainName, 0093 int result) 0094 { 0095 // see gotPersistentUserReply for values stored and why 0096 switch (type) { 0097 case KIO::AskUserActionInterface::QuestionTwoActions: 0098 case KIO::AskUserActionInterface::QuestionTwoActionsCancel: 0099 case KIO::AskUserActionInterface::WarningTwoActions: 0100 case KIO::AskUserActionInterface::WarningTwoActionsCancel: 0101 cg.writeEntry(dontAskAgainName, result == KIO::WorkerBase::PrimaryAction); 0102 cg.sync(); 0103 break; 0104 case KIO::AskUserActionInterface::WarningContinueCancel: 0105 cg.writeEntry(dontAskAgainName, false); 0106 cg.sync(); 0107 break; 0108 default: 0109 break; 0110 } 0111 } 0112 0113 QWidget *KIO::WidgetsAskUserActionHandlerPrivate::getParentWidget(KJob *job) 0114 { 0115 // This needs to be in qpointer, otherwise copying process 0116 // will crash if done in background and dolphin is closed 0117 QPointer<QWidget> parentWidget = nullptr; 0118 0119 if (job) { 0120 parentWidget = KJobWidgets::window(job); 0121 } 0122 0123 return getParentWidget(parentWidget); 0124 } 0125 0126 QWidget *KIO::WidgetsAskUserActionHandlerPrivate::getParentWidget(QWidget *widget) 0127 { 0128 // This needs to be in qpointer, otherwise copying process 0129 // will crash if done in background and dolphin is closed 0130 QPointer<QWidget> parentWidget = widget; 0131 0132 if (!parentWidget) { 0133 parentWidget = this->m_parentWidget; 0134 } 0135 0136 if (!parentWidget) { 0137 parentWidget = qApp->activeWindow(); 0138 } 0139 0140 return parentWidget; 0141 } 0142 0143 KIO::WidgetsAskUserActionHandler::WidgetsAskUserActionHandler(QObject *parent) 0144 : KIO::AskUserActionInterface(parent) 0145 , d(new WidgetsAskUserActionHandlerPrivate(this)) 0146 { 0147 } 0148 0149 KIO::WidgetsAskUserActionHandler::~WidgetsAskUserActionHandler() 0150 { 0151 } 0152 0153 void KIO::WidgetsAskUserActionHandler::askUserRename(KJob *job, 0154 const QString &title, 0155 const QUrl &src, 0156 const QUrl &dest, 0157 KIO::RenameDialog_Options options, 0158 KIO::filesize_t sizeSrc, 0159 KIO::filesize_t sizeDest, 0160 const QDateTime &ctimeSrc, 0161 const QDateTime &ctimeDest, 0162 const QDateTime &mtimeSrc, 0163 const QDateTime &mtimeDest) 0164 { 0165 QMetaObject::invokeMethod(qGuiApp, [=] { 0166 auto *dlg = new KIO::RenameDialog(d->getParentWidget(job), title, src, dest, options, sizeSrc, sizeDest, ctimeSrc, ctimeDest, mtimeSrc, mtimeDest); 0167 0168 dlg->setAttribute(Qt::WA_DeleteOnClose); 0169 dlg->setWindowModality(Qt::WindowModal); 0170 0171 connect(job, &KJob::finished, dlg, &QDialog::reject); 0172 connect(dlg, &QDialog::finished, this, [this, job, dlg](const int exitCode) { 0173 KIO::RenameDialog_Result result = static_cast<RenameDialog_Result>(exitCode); 0174 const QUrl newUrl = result == Result_AutoRename ? dlg->autoDestUrl() : dlg->newDestUrl(); 0175 Q_EMIT askUserRenameResult(result, newUrl, job); 0176 }); 0177 0178 dlg->show(); 0179 }); 0180 } 0181 0182 void KIO::WidgetsAskUserActionHandler::askUserSkip(KJob *job, KIO::SkipDialog_Options options, const QString &errorText) 0183 { 0184 QMetaObject::invokeMethod(qGuiApp, [=] { 0185 auto *dlg = new KIO::SkipDialog(d->getParentWidget(job), options, errorText); 0186 dlg->setAttribute(Qt::WA_DeleteOnClose); 0187 dlg->setWindowModality(Qt::WindowModal); 0188 0189 connect(job, &KJob::finished, dlg, &QDialog::reject); 0190 connect(dlg, &QDialog::finished, this, [this, job](const int exitCode) { 0191 Q_EMIT askUserSkipResult(static_cast<KIO::SkipDialog_Result>(exitCode), job); 0192 }); 0193 0194 dlg->show(); 0195 }); 0196 } 0197 0198 struct ProcessAskDeleteResult { 0199 QStringList prettyList; 0200 KMessageDialog::Type dialogType = KMessageDialog::QuestionTwoActions; 0201 KGuiItem acceptButton; 0202 QString text; 0203 QIcon icon; 0204 QString title = i18n("Delete Permanently"); 0205 bool isSingleUrl = false; 0206 }; 0207 0208 using AskIface = KIO::AskUserActionInterface; 0209 static ProcessAskDeleteResult processAskDelete(const QList<QUrl> &urls, AskIface::DeletionType deletionType) 0210 { 0211 ProcessAskDeleteResult res; 0212 res.prettyList.reserve(urls.size()); 0213 std::transform(urls.cbegin(), urls.cend(), std::back_inserter(res.prettyList), [](const auto &url) { 0214 if (url.scheme() == QLatin1String("trash")) { 0215 QString path = url.path(); 0216 // HACK (#98983): remove "0-foo". Note that it works better than 0217 // displaying KFileItem::name(), for files under a subdir. 0218 static const QRegularExpression re(QStringLiteral("^/[0-9]+-")); 0219 path.remove(re); 0220 return path; 0221 } else { 0222 return url.toDisplayString(QUrl::PreferLocalFile); 0223 } 0224 }); 0225 0226 const int urlCount = res.prettyList.size(); 0227 res.isSingleUrl = urlCount == 1; 0228 0229 switch (deletionType) { 0230 case AskIface::Delete: { 0231 res.dialogType = KMessageDialog::QuestionTwoActions; // Using Question* so the Delete button is pre-selected. Bug 462845 0232 res.icon = QIcon::fromTheme(QStringLiteral("dialog-warning")); 0233 if (res.isSingleUrl) { 0234 res.text = xi18nc("@info", 0235 "Do you really want to permanently delete this item?<nl/><nl/>" 0236 "<filename>%1</filename><nl/><nl/>" 0237 "<emphasis strong='true'>This action cannot be undone.</emphasis>", 0238 res.prettyList.at(0)); 0239 } else { 0240 res.text = xi18ncp("@info", 0241 "Do you really want to permanently delete this %1 item?<nl/><nl/>" 0242 "<emphasis strong='true'>This action cannot be undone.</emphasis>", 0243 "Do you really want to permanently delete these %1 items?<nl/><nl/>" 0244 "<emphasis strong='true'>This action cannot be undone.</emphasis>", 0245 urlCount); 0246 } 0247 res.acceptButton = KGuiItem(i18nc("@action:button", "Delete Permanently"), QStringLiteral("edit-delete")); 0248 break; 0249 } 0250 case AskIface::DeleteInsteadOfTrash: { 0251 res.dialogType = KMessageDialog::WarningTwoActions; 0252 if (res.isSingleUrl) { 0253 res.text = xi18nc("@info", 0254 "Moving this item to Trash failed as it is too large." 0255 " Permanently delete it instead?<nl/><nl/>" 0256 "<filename>%1</filename><nl/><nl/>" 0257 "<emphasis strong='true'>This action cannot be undone.</emphasis>", 0258 res.prettyList.at(0)); 0259 } else { 0260 res.text = xi18ncp("@info", 0261 "Moving this %1 item to Trash failed as it is too large." 0262 " Permanently delete it instead?<nl/>" 0263 "<emphasis strong='true'>This action cannot be undone.</emphasis>", 0264 "Moving these %1 items to Trash failed as they are too large." 0265 " Permanently delete them instead?<nl/><nl/>" 0266 "<emphasis strong='true'>This action cannot be undone.</emphasis>", 0267 urlCount); 0268 } 0269 res.acceptButton = KGuiItem(i18nc("@action:button", "Delete Permanently"), QStringLiteral("edit-delete")); 0270 break; 0271 } 0272 case AskIface::EmptyTrash: { 0273 res.dialogType = KMessageDialog::QuestionTwoActions; // Using Question* so the Delete button is pre-selected. 0274 res.icon = QIcon::fromTheme(QStringLiteral("dialog-warning")); 0275 res.text = xi18nc("@info", 0276 "Do you want to permanently delete all items from the Trash?<nl/><nl/>" 0277 "<emphasis strong='true'>This action cannot be undone.</emphasis>"); 0278 res.acceptButton = KGuiItem(i18nc("@action:button", "Empty Trash"), QStringLiteral("user-trash")); 0279 break; 0280 } 0281 case AskIface::Trash: { 0282 if (res.isSingleUrl) { 0283 res.text = xi18nc("@info", 0284 "Do you really want to move this item to the Trash?<nl/>" 0285 "<filename>%1</filename>", 0286 res.prettyList.at(0)); 0287 } else { 0288 res.text = 0289 xi18ncp("@info", "Do you really want to move this %1 item to the Trash?", "Do you really want to move these %1 items to the Trash?", urlCount); 0290 } 0291 res.title = i18n("Move to Trash"); 0292 res.acceptButton = KGuiItem(res.title, QStringLiteral("user-trash")); 0293 break; 0294 } 0295 default: 0296 break; 0297 } 0298 return res; 0299 } 0300 0301 void KIO::WidgetsAskUserActionHandler::askUserDelete(const QList<QUrl> &urls, DeletionType deletionType, ConfirmationType confirmationType, QWidget *parent) 0302 { 0303 QString keyName; 0304 bool ask = (confirmationType == ForceConfirmation); 0305 if (!ask) { 0306 // The default value for confirmations is true for delete and false 0307 // for trash. If you change this, please also update: 0308 // dolphin/src/settings/general/confirmationssettingspage.cpp 0309 bool defaultValue = true; 0310 0311 switch (deletionType) { 0312 case DeleteInsteadOfTrash: 0313 case Delete: 0314 keyName = QStringLiteral("ConfirmDelete"); 0315 break; 0316 case Trash: 0317 keyName = QStringLiteral("ConfirmTrash"); 0318 defaultValue = false; 0319 break; 0320 case EmptyTrash: 0321 keyName = QStringLiteral("ConfirmEmptyTrash"); 0322 break; 0323 } 0324 0325 KSharedConfigPtr kioConfig = KSharedConfig::openConfig(QStringLiteral("kiorc"), KConfig::NoGlobals); 0326 ask = kioConfig->group(QStringLiteral("Confirmations")).readEntry(keyName, defaultValue); 0327 } 0328 0329 if (!ask) { 0330 Q_EMIT askUserDeleteResult(true, urls, deletionType, parent); 0331 return; 0332 } 0333 0334 QMetaObject::invokeMethod(qGuiApp, [=] { 0335 const auto &[prettyList, dialogType, acceptButton, text, icon, title, singleUrl] = processAskDelete(urls, deletionType); 0336 KMessageDialog *dlg = new KMessageDialog(dialogType, text, parent); 0337 dlg->setAttribute(Qt::WA_DeleteOnClose); 0338 dlg->setCaption(title); 0339 dlg->setIcon(icon); 0340 dlg->setButtons(acceptButton, KStandardGuiItem::cancel()); 0341 if (!singleUrl) { 0342 dlg->setListWidgetItems(prettyList); 0343 } 0344 dlg->setDontAskAgainText(i18nc("@option:checkbox", "Do not ask again")); 0345 // If we get here, !ask must be false 0346 dlg->setDontAskAgainChecked(!ask); 0347 0348 connect(dlg, &QDialog::finished, this, [=](const int buttonCode) { 0349 const bool isDelete = (buttonCode == KMessageDialog::PrimaryAction); 0350 0351 Q_EMIT askUserDeleteResult(isDelete, urls, deletionType, parent); 0352 0353 if (isDelete) { 0354 KSharedConfigPtr kioConfig = KSharedConfig::openConfig(QStringLiteral("kiorc"), KConfig::NoGlobals); 0355 KConfigGroup cg = kioConfig->group(QStringLiteral("Confirmations")); 0356 cg.writeEntry(keyName, !dlg->isDontAskAgainChecked()); 0357 cg.sync(); 0358 } 0359 }); 0360 0361 dlg->setWindowModality(Qt::WindowModal); 0362 dlg->show(); 0363 }); 0364 } 0365 0366 void KIO::WidgetsAskUserActionHandler::requestUserMessageBox(MessageDialogType type, 0367 const QString &text, 0368 const QString &title, 0369 const QString &primaryActionText, 0370 const QString &secondaryActionText, 0371 const QString &primaryActionIconName, 0372 const QString &secondaryActionIconName, 0373 const QString &dontAskAgainName, 0374 const QString &details, 0375 QWidget *parent) 0376 { 0377 if (d->gotPersistentUserReply(type, 0378 KSharedConfig::openConfig(QStringLiteral("kioslaverc"))->group(QStringLiteral("Notification Messages")), 0379 dontAskAgainName)) { 0380 return; 0381 } 0382 0383 const KGuiItem primaryActionButton(primaryActionText, primaryActionIconName); 0384 const KGuiItem secondaryActionButton(secondaryActionText, secondaryActionIconName); 0385 0386 // It's "Do not ask again" every where except with Information 0387 QString dontAskAgainText = i18nc("@option:check", "Do not ask again"); 0388 0389 KMessageDialog::Type dlgType; 0390 bool hasCancelButton = false; 0391 0392 switch (type) { 0393 case AskUserActionInterface::QuestionTwoActions: 0394 dlgType = KMessageDialog::QuestionTwoActions; 0395 break; 0396 case AskUserActionInterface::QuestionTwoActionsCancel: 0397 dlgType = KMessageDialog::QuestionTwoActionsCancel; 0398 hasCancelButton = true; 0399 break; 0400 case AskUserActionInterface::WarningTwoActions: 0401 dlgType = KMessageDialog::WarningTwoActions; 0402 break; 0403 case AskUserActionInterface::WarningTwoActionsCancel: 0404 dlgType = KMessageDialog::WarningTwoActionsCancel; 0405 hasCancelButton = true; 0406 break; 0407 case AskUserActionInterface::WarningContinueCancel: 0408 dlgType = KMessageDialog::WarningContinueCancel; 0409 hasCancelButton = true; 0410 break; 0411 case AskUserActionInterface::Information: 0412 dlgType = KMessageDialog::Information; 0413 dontAskAgainText = i18nc("@option:check", "Do not show this message again"); 0414 break; 0415 case AskUserActionInterface::Error: 0416 dlgType = KMessageDialog::Error; 0417 dontAskAgainText = QString{}; // No dontAskAgain checkbox 0418 break; 0419 default: 0420 qCWarning(KIO_WIDGETS) << "Unknown message dialog type" << type; 0421 return; 0422 } 0423 0424 QMetaObject::invokeMethod(qGuiApp, [=]() { 0425 auto cancelButton = hasCancelButton ? KStandardGuiItem::cancel() : KGuiItem(); 0426 auto *dialog = new KMessageDialog(dlgType, text, d->getParentWidget(parent)); 0427 0428 dialog->setAttribute(Qt::WA_DeleteOnClose); 0429 dialog->setCaption(title); 0430 dialog->setIcon(QIcon{}); 0431 dialog->setButtons(primaryActionButton, secondaryActionButton, cancelButton); 0432 dialog->setDetails(details); 0433 dialog->setDontAskAgainText(dontAskAgainText); 0434 dialog->setDontAskAgainChecked(false); 0435 dialog->setOpenExternalLinks(true); // Allow opening external links in the text labels 0436 0437 connect(dialog, &QDialog::finished, this, [=](const int result) { 0438 KIO::WorkerBase::ButtonCode btnCode; 0439 switch (result) { 0440 case KMessageDialog::PrimaryAction: 0441 if (dlgType == KMessageDialog::WarningContinueCancel) { 0442 btnCode = KIO::WorkerBase::Continue; 0443 } else { 0444 btnCode = KIO::WorkerBase::PrimaryAction; 0445 } 0446 break; 0447 case KMessageDialog::SecondaryAction: 0448 btnCode = KIO::WorkerBase::SecondaryAction; 0449 break; 0450 case KMessageDialog::Cancel: 0451 btnCode = KIO::WorkerBase::Cancel; 0452 break; 0453 case KMessageDialog::Ok: 0454 btnCode = KIO::WorkerBase::Ok; 0455 break; 0456 default: 0457 qCWarning(KIO_WIDGETS) << "Unknown message dialog result" << result; 0458 return; 0459 } 0460 0461 Q_EMIT messageBoxResult(btnCode); 0462 0463 if ((result != KMessageDialog::Cancel) && dialog->isDontAskAgainChecked()) { 0464 KSharedConfigPtr reqMsgConfig = KSharedConfig::openConfig(QStringLiteral("kioslaverc")); 0465 KConfigGroup cg = reqMsgConfig->group(QStringLiteral("Notification Messages")); 0466 d->savePersistentUserReply(type, cg, dontAskAgainName, result); 0467 } 0468 }); 0469 0470 dialog->show(); 0471 }); 0472 } 0473 0474 void KIO::WidgetsAskUserActionHandler::setWindow(QWidget *window) 0475 { 0476 d->m_parentWidget = window; 0477 } 0478 0479 void KIO::WidgetsAskUserActionHandler::askIgnoreSslErrors(const QVariantMap &sslErrorData, QWidget *parent) 0480 { 0481 QWidget *parentWidget = d->getParentWidget(parent); 0482 0483 QString message = i18n("The server failed the authenticity check (%1).\n\n", sslErrorData[QLatin1String("hostname")].toString()); 0484 0485 message += sslErrorData[QLatin1String("sslError")].toString(); 0486 0487 auto *dialog = new KMessageDialog(KMessageDialog::WarningTwoActionsCancel, message, parentWidget); 0488 0489 dialog->setAttribute(Qt::WA_DeleteOnClose); 0490 dialog->setCaption(i18n("Server Authentication")); 0491 dialog->setIcon(QIcon{}); 0492 dialog->setButtons(KGuiItem{i18n("&Details"), QStringLiteral("documentinfo")}, KStandardGuiItem::cont(), KStandardGuiItem::cancel()); 0493 0494 connect(dialog, &KMessageDialog::finished, this, [this, parentWidget, sslErrorData](int result) { 0495 if (result == KMessageDialog::PrimaryAction) { 0496 showSslDetails(sslErrorData, parentWidget); 0497 } else if (result == KMessageDialog::SecondaryAction) { 0498 // continue(); 0499 Q_EMIT askIgnoreSslErrorsResult(1); 0500 } else if (result == KMessageDialog::Cancel) { 0501 // cancel(); 0502 Q_EMIT askIgnoreSslErrorsResult(0); 0503 } else { 0504 Q_UNREACHABLE(); 0505 } 0506 }); 0507 0508 dialog->show(); 0509 } 0510 0511 void KIO::WidgetsAskUserActionHandler::showSslDetails(const QVariantMap &sslErrorData, QWidget *parentWidget) 0512 { 0513 const QStringList sslList = sslErrorData[QLatin1String("peerCertChain")].toStringList(); 0514 0515 QList<QSslCertificate> certChain; 0516 bool decodedOk = true; 0517 for (const QString &str : sslList) { 0518 certChain.append(QSslCertificate(str.toUtf8())); 0519 if (certChain.last().isNull()) { 0520 decodedOk = false; 0521 break; 0522 } 0523 } 0524 0525 QMetaObject::invokeMethod(qGuiApp, [=] { 0526 if (decodedOk) { // Use KSslInfoDialog 0527 KSslInfoDialog *ksslDlg = new KSslInfoDialog(parentWidget); 0528 ksslDlg->setSslInfo( 0529 certChain, 0530 QString(), 0531 sslErrorData[QLatin1String("hostname")].toString(), 0532 sslErrorData[QLatin1String("protocol")].toString(), 0533 sslErrorData[QLatin1String("cipher")].toString(), 0534 sslErrorData[QLatin1String("usedBits")].toInt(), 0535 sslErrorData[QLatin1String("bits")].toInt(), 0536 KSslInfoDialog::certificateErrorsFromString(sslErrorData[QLatin1String("certificateErrors")].toStringList().join(QLatin1Char('\n')))); 0537 0538 // KSslInfoDialog deletes itself by setting Qt::WA_DeleteOnClose 0539 0540 QObject::connect(ksslDlg, &QDialog::finished, this, [this, sslErrorData, parentWidget]() { 0541 // KSslInfoDialog only has one button, QDialogButtonBox::Close 0542 askIgnoreSslErrors(sslErrorData, parentWidget); 0543 }); 0544 0545 ksslDlg->show(); 0546 return; 0547 } 0548 0549 // Fallback to a generic message box 0550 auto *dialog = new KMessageDialog(KMessageDialog::Information, i18n("The peer SSL certificate chain appears to be corrupt."), parentWidget); 0551 0552 dialog->setAttribute(Qt::WA_DeleteOnClose); 0553 dialog->setCaption(i18n("SSL")); 0554 dialog->setButtons(KStandardGuiItem::ok()); 0555 0556 QObject::connect(dialog, &QDialog::finished, this, [this](const int result) { 0557 Q_EMIT askIgnoreSslErrorsResult(result == KMessageDialog::Ok ? 1 : 0); 0558 }); 0559 0560 dialog->show(); 0561 }); 0562 } 0563 0564 #include "moc_widgetsaskuseractionhandler.cpp"