按字母顺序更新和排序Qt ComboBox

Update and sort Qt ComboBoxes alphabetically

本文关键字:排序 Qt ComboBox 更新 顺序      更新时间:2023-10-16

我想写一个程序,像Qt框架中的一个小导航系统一样工作,但我对它仍然很陌生。

我创建了一个包含两个组合框的对话框。每个组合框都包含所有"城市名称"。在初始化时,两个框的内容按字母顺序排序。

如果选择了第一个框中的名称,则不应以第二个框和另一个方式显示该名称。

如果选择了另一个项目,我成功删除了该项目并再次添加它,但知道我无法再对它们进行排序

这是我到目前为止尝试更新的内容:

for(std::vector<City>::iterator iter = citylist.begin(); iter != citylist.end(); iter++){
    if(ui->combo2->currentText() != (*iter).getName()
            and ui->combo1->findText((*iter).getName()) == -1){
        ui->combo1->addItem((*iter).getName(),QComboBox::InsertAlphabetically);
    }
}

但它不会按字母顺序插入项目...

所以我之后尝试对其进行排序:

 QSortFilterProxyModel* proxy = new QSortFilterProxyModel(ui->combo1);
proxy->setSourceModel(ui->combo1->model());
// combo's current model must be reparented,
// otherwise QComboBox::setModel() will delete it
ui->combo1->model()->setParent(proxy);
ui->combo1->setModel(proxy);
// sort
ui->combo1->model()->sort(0);

但是,如果我尝试调用此函数,则会发生错误并且应用程序将终止。

那么,有没有人能帮助我呢?

你快到了!

ui->comboBox1.addItem("myitem");
// qApp->processEvents();  not really needed
ui->comboBox1.model()->sort(0);

您正在尝试使用 QComboBox 的内部模型作为proxy的源模型。这是行不通的,因为QComboBox拥有其内部模型,当您调用 QComboBox::setModel 时,以前的模型将被删除(尽管您重置了其父模型)。您需要创建单独的源模型。方便的是,如果城市列表相同,则可以对两个组合框使用一个源模型。

使用 QSortFilterProxyModel 进行排序很容易,但要用它排除一个特定的字符串却非常困难。您可以子类QSortFilterProxyModel::filterAcceptsRow并实现所需的行为。我决定用一点黑魔法代替(见这个答案)。

私有类字段:

private:
  QSortFilterProxyModel *proxy1, *proxy2;

源:

MainWindow::MainWindow(QWidget *parent) :
  QMainWindow(parent),
  ui(new Ui::MainWindow)
{
  ui->setupUi(this);
  QStandardItemModel* model = new QStandardItemModel(this);
  foreach(QString name, QStringList()
      << "Paris"<< "London"<< "Moscow" << "Tokyo" << "Berlin" << "Amsterdam") {
    model->appendRow(new QStandardItem(name));
  }
  proxy1 = new QSortFilterProxyModel();
  proxy1->setSourceModel(model);
  proxy1->sort(0);
  ui->comboBox1->setModel(proxy1);
  proxy2 = new QSortFilterProxyModel();
  proxy2->setSourceModel(model);
  proxy2->sort(0);
  ui->comboBox2->setModel(proxy2);
  connect(ui->comboBox1, &QComboBox::currentTextChanged,
          this, &MainWindow::something_changed);
  connect(ui->comboBox2, &QComboBox::currentTextChanged,
          this, &MainWindow::something_changed);
  something_changed();
}
void MainWindow::something_changed() {
  ui->comboBox1->blockSignals(true); //prevent recursion
  ui->comboBox2->blockSignals(true);
  proxy2->setFilterRegExp(QString("^(?!(%1)$)").arg(
                          QRegExp::escape(ui->comboBox1->currentText())));
  proxy1->setFilterRegExp(QString("^(?!(%1)$)").arg(
                          QRegExp::escape(ui->comboBox2->currentText())));
  ui->comboBox1->blockSignals(false);
  ui->comboBox2->blockSignals(false);
}

在Qt 5.3中测试。