- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
如何防止 QDialog 类在按下“确定”按钮后关闭?仅当在此对话框中正确执行了一些操作时我才需要关闭窗口,在其他情况下我不需要关闭此窗口。
最佳答案
一般来说,欺骗用户是一种坏习惯。如果一个按钮没有被禁用,那么当用户点击它时它会更好地工作。
因此,显而易见的解决方案是在满足必要的先决条件之前禁用该按钮。对于完成对话框的按钮,您应该使用 QDialogButtonBox
而不是离散按钮,因为在不同的平台上,这些按钮将根据按钮的角色/类型在框中以不同方式排列。
下面是如何完成的示例。适用于 Qt 4 和 5。
已注意代码与现有样式表的互操作。
// https://github.com/KubaO/stackoverflown/tree/master/questions/buttonbox-22404318
#include <QtGui>
#if QT_VERSION >= QT_VERSION_CHECK(5,0,0)
#include <QtWidgets>
#endif
#include <type_traits>
首先,让我们使用一些样式表操作助手:
void styleSheetSet(QWidget *w, const QString &what) {
auto const token = QStringLiteral("/*>*/%1/*<*/").arg(what);
if (what.isEmpty() || w->styleSheet().contains(token)) return;
w->setStyleSheet(w->styleSheet().append(token));
}
void styleSheetClear(QWidget *w, const QString &what) {
const auto token = QStringLiteral("/*>*/%1/*<*/").arg(what);
if (what.isEmpty() || ! w->styleSheet().contains(token)) return;
w->setStyleSheet(w->styleSheet().remove(token));
}
void styleSheetSelect(QWidget *w, bool selector,
const QString & onTrue,
const QString & onFalse = {})
{
styleSheetSet(w, selector ? onTrue : onFalse);
styleSheetClear(w, selector ? onFalse : onTrue);
}
template <typename T, typename U>
void setSelect(QSet<T> &set, bool b, const U &val) {
if (b) set.insert(val); else set.remove(val);
}
和递归父搜索:
bool hasParent(QObject *obj, QObject *const parent) {
Q_ASSERT(obj);
while (obj = obj->parent())
if (obj == parent) return true;
return obj == parent;
}
DialogValidator
管理单个对话框的验证器。首先,当 QLineEdit
和通用小部件的内容发生变化时调用的插槽:
class DialogValidator : public QObject {
Q_OBJECT
QSet<QWidget*> m_validWidgets;
int m_needsValid = 0;
Q_SLOT void checkWidget() {
if (sender()->isWidgetType())
checkValidity(static_cast<QWidget*>(sender()));
}
Q_SLOT void checkLineEdit() {
if (auto *l = qobject_cast<QLineEdit*>(sender()))
checkValidity(l);
}
void checkValidity(QLineEdit *l) {
indicateValidity(l, l->hasAcceptableInput());
}
void checkValidity(QWidget *w) {
auto validator = w->findChild<QValidator*>();
if (!validator) return;
auto prop = w->metaObject()->userProperty();
QVariant value = prop.read(w);
int pos;
QString text = value.toString();
bool isValid =
validator->validate(text, pos) == QValidator::Acceptable;
indicateValidity(w, isValid);
}
void indicateValidity(QWidget *w, bool isValid) {
auto *combo = qobject_cast<QComboBox*>(w->parentWidget());
setSelect(m_validWidgets, isValid, combo ? combo : w);
styleSheetSelect(w, !isValid,
QStringLiteral("%1 { background: yellow }")
.arg(QLatin1String(w->metaObject()->className())));
emit newValidity(m_validWidgets.count() == m_needsValid);
}
使用 add
方法将验证器添加到对话框验证器。如果我们希望对动态类型的小部件进行特殊处理,则应使用 addPoly
方法 - 如果有的话,它将分派(dispatch)到特定类型的重载:
template<typename W>
typename std::enable_if<!std::is_same<QWidget,W>::value, bool>::type
addPoly(W* w, QValidator *v) {
if (!w) return false;
return (add(w,v), true);
}
public:
DialogValidator(QObject *parent = {}) : QObject(parent) {}
Q_SIGNAL void newValidity(bool);
void addPoly(QWidget *w, QValidator *v) {
addPoly(qobject_cast<QLineEdit*>(w), v) ||
addPoly(qobject_cast<QComboBox*>(w), v) ||
(add(w, v), true);
}
然后,静态类型的 add
方法:
void add(QComboBox *b, QValidator *v) {
if (auto *l = b->lineEdit())
add(l, v);
}
void add(QLineEdit *l, QValidator *v) {
l->setValidator(v);
connect(l, SIGNAL(textChanged(QString)), SLOT(checkLineEdit()));
m_needsValid++;
checkValidity(l);
}
void add(QWidget *w, QValidator *v) {
Q_ASSERT(hasParent(v, w));
auto prop = w->metaObject()->userProperty();
auto propChanged = prop.notifySignal();
static auto check = metaObject()->method(metaObject()->indexOfSlot("checkWidget()"));
Q_ASSERT(check.isValid());
if (!prop.isValid() || !propChanged.isValid())
return qWarning("DialogValidator::add: The target widget has no user property with a notifier.");
if (connect(w, propChanged, this, check)) {
m_needsValid++;
checkValidity(w);
}
}
最后,构造验证器的便捷方法:
template <typename V, typename W, typename...Args>
typename std::enable_if<
std::is_base_of<QWidget, W>::value && std::is_base_of<QValidator, V>::value, V*>::type
add(W *w, Args...args) {
V *validator = new V(std::forward<Args>(args)..., w);
return add(w, validator), validator;
}
};
我们的验证对话:
class MyDialog : public QDialog {
Q_OBJECT
QFormLayout m_layout{this};
QLineEdit m_cFactor;
QLineEdit m_dFactor;
QDialogButtonBox m_buttons{QDialogButtonBox::Ok | QDialogButtonBox::Cancel};
DialogValidator m_validator;
public:
MyDialog(QWidget *parent = {}, Qt::WindowFlags f = {}) : QDialog(parent, f) {
m_layout.addRow("Combobulation Factor", &m_cFactor);
m_layout.addRow("Decombobulation Factor", &m_dFactor);
m_layout.addRow(&m_buttons);
connect(&m_buttons, SIGNAL(accepted()), SLOT(accept()));
connect(&m_buttons, SIGNAL(rejected()), SLOT(reject()));
connect(&m_validator, SIGNAL(newValidity(bool)),
m_buttons.button(QDialogButtonBox::Ok), SLOT(setEnabled(bool)));
// QLineEdit-specific validator
m_validator.add<QIntValidator>(&m_cFactor, 0, 50);
// Generic user property-based validator
m_validator.add<QIntValidator, QWidget>(&m_dFactor, -50, 0);
}
};
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
MyDialog d;
d.show();
return a.exec();
}
#include "main.moc"
关于c++ - 如何防止 QDialog 类关闭,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22404318/
我正在使用的网站上有一个非 Canvas 导航。关闭 Canvas 导航的默认状态是关闭的,这在移动网站上运行良好,因为您可以打开它并选择您的链接,但在桌面上关闭它并打开它会隐藏用户的信息,我希望它是
我有一个 NSViewController 是这样连接的: 在底部 viewController 中,我尝试使用 self.dismiss(self) 关闭它,但是,它会产生此错误: [General
我昨天制作了一个扩展的 JQuery 搜索框,它的作用就像一个魅力!但是,我在创建一个脚本时遇到问题,当用户单击搜索框时,它会关闭。 这是我的 JQuery: function expandSearc
我一辈子都无法在 API V3 中一次只显示一个信息窗口。我需要一个在下一次开放之前关闭。还希望在 map 上的任何地方关闭 infoWindow onclick。这是否在初始化函数中? 这是我的完整
关闭和清理套接字的正确方法是什么? 我在辅助线程中运行 io_service,我需要关闭与主线程的连接: void closeConnection() { ioc.post([&socket]
我的 Selenium 测试看起来像这样:客户选择金融产品,填写一些必要的数据,并在打印预览中显示条款/协议(protocol)文档(根据本地法律的要求)。打印/关闭打印预览对话框后,客户输入更多数据
我目前正在从 android 网站了解 Navigation Drawer,我正在使用他们的示例 http://developer.android.com/training/implementing-
尝试通过 expo 在模拟器上运行 react-native 应用程序时出现此错误。 Couldn't start project on Android: Error running adb: adb
方法一 function transform(ar) { var alStr = []; for(var i=0; i
我想按以下方式自定义我的抽屉导航: 我希望在抽屉打开时显示一个图标,在抽屉关闭时显示另一个图标,而不是将菜单图标稍微向左滑动的当前默认动画。 关于我在哪里可以找到类似内容的任何想法/线索? 我做了一些
我们刚刚从 0.6.2 或 0.7 升级了我们的 dropwizard 版本,发现 .yml 文件中的很多配置都发生了变化。尽管我们能够弄清楚其中的大部分,但我们无法弄清楚如何关闭“requestLo
从 celery 2.4.5 升级后,我开始让 celery 随机关闭。 我在 centOS 机器上使用 celery 3.0.12、boto 2.6 和 amazon sqs 和 django 1.
我试图包含一些语句来指导用户更多地了解文件无法打开或关闭的原因。文件在写入模式下无法打开的一些可能情况是什么?无法关闭怎么办? FILE *fp; if(!(fp = fopen("testing",
我有一个DLL,可以访问数据库并从存储在配置文件中的应用程序设置中读取连接字符串。然后,引用此DLL的应用程序将需要在其配置文件中为此配置设置设置值。 我遇到的问题是,生成的配置代码会通过Defaul
我将 UIDatePicker 添加为 UITextField 的输入 View UIDatePicker *oBirth; NSDateFormatter *dateFormat; _edit
我有以下代码: SecondViewController *secondView = [[SecondViewController alloc] initWithNibName:@"SecondVie
Closed. This question does not meet Stack Overflow guidelines。它当前不接受答案。 想改善这个问题吗?更新问题,以便将其作为on-topic
通常,按下 option 键关闭窗口会关闭应用程序中的所有窗口。在我的应用程序中,我希望它仅关闭与用户正在关闭的窗口相关的窗口。我怎样才能做到这一点?我可以为所有窗口实现 windowShouldCl
我有一个 NSWindow,它托管一个已连接到脚本处理程序的 WebView。 现在,当用户单击 WebView 上的控件上的按钮时,它会调用我的对象上的 Objective C 方法。 在这种特定情
我想根据 MBP 上的相机使用情况自动化个人工作流程。 基本上我想知道是否任何 的摄像头(内置或 USB)已打开或关闭,因此我可以运行我将创建的程序或脚本。 我认为如果我需要轮询相机状态也可以,但基于
我是一名优秀的程序员,十分优秀!