#include "EditTableDialog.h"
#include "Settings.h"
#include "ForeignKeyEditorDelegate.h"
#include "ui_EditTableDialog.h"
#include "sqlitetablemodel.h"
#include "sqlitedb.h"
#include "SelectItemsPopup.h"
#include
#include
#include
#include
#include
#include
#include
Q_DECLARE_METATYPE(std::shared_ptr<:uniqueconstraint>)
Q_DECLARE_METATYPE(std::shared_ptr<:foreignkeyclause>)
Q_DECLARE_METATYPE(std::shared_ptr<:checkconstraint>)
Q_DECLARE_METATYPE(sqlb::StringVector)
Q_DECLARE_METATYPE(sqlb::IndexedColumnVector)
// Styled Item Delegate for non-editable columns
class NoEditDelegate: public QStyledItemDelegate {
public:
explicit NoEditDelegate(QObject* parent=nullptr): QStyledItemDelegate(parent) {}
QWidget* createEditor(QWidget* /* parent */, const QStyleOptionViewItem& /* option */, const QModelIndex& /* index */) const override {
return nullptr;
}
};
EditTableDialog::EditTableDialog(DBBrowserDB& db, const sqlb::ObjectIdentifier& tableName, bool createTable, QWidget* parent)
: QDialog(parent),
ui(new Ui::EditTableDialog),
pdb(db),
curTable(tableName),
m_table(tableName.name()),
m_bNewTable(createTable),
m_sRestorePointName(pdb.generateSavepointName("edittable"))
{
// Create UI
ui->setupUi(this);
ui->widgetExtension->setVisible(false);
connect(ui->treeWidget, &QTreeWidget::itemChanged, this, &EditTableDialog::fieldItemChanged);
connect(ui->tableIndexConstraints, &QTableWidget::itemChanged, this, &EditTableDialog::indexConstraintItemChanged);
connect(ui->tableForeignKeys, &QTableWidget::itemChanged, this, &EditTableDialog::foreignKeyItemChanged);
connect(ui->tableCheckConstraints, &QTableWidget::itemChanged, this, &EditTableDialog::checkConstraintItemChanged);
// Set item delegate for foreign key column
ui->treeWidget->setItemDelegateForColumn(kForeignKey, new ForeignKeyEditorDelegate(db, m_table, this));
ui->tableForeignKeys->setItemDelegateForColumn(kForeignKeyReferences, new ForeignKeyEditorDelegate(db, m_table, this));
// Disallow edition of checkable columns or columns with combo-boxes
ui->treeWidget->setItemDelegateForColumn(kType, new NoEditDelegate(this));
ui->treeWidget->setItemDelegateForColumn(kNotNull, new NoEditDelegate(this));
ui->treeWidget->setItemDelegateForColumn(kPrimaryKey, new NoEditDelegate(this));
ui->treeWidget->setItemDelegateForColumn(kAutoIncrement, new NoEditDelegate(this));
ui->treeWidget->setItemDelegateForColumn(kUnique, new NoEditDelegate(this));
ui->treeWidget->setItemDelegateForColumn(kCollation, new NoEditDelegate(this));
// Set up popup menu for adding constraints
QMenu* constraint_menu = new QMenu(this);
constraint_menu->addAction(ui->actionAddPrimaryKey);
constraint_menu->addAction(ui->actionAddUniqueConstraint);
connect(ui->actionAddPrimaryKey, &QAction::triggered, this, [this]() { addIndexConstraint(true); });
connect(ui->actionAddUniqueConstraint, &QAction::triggered, this, [this]() { addIndexConstraint(false); });
ui->buttonAddIndexConstraint->setMenu(constraint_menu);
// Get list of all collations
db.executeSQL("PRAGMA collation_list;", false, true, [this](int column_count, std::vector columns, std::vector) -> bool {
if(column_count >= 2)
m_collationList.push_back(columns.at(1));
return false;
});
if(!m_collationList.contains(""))
m_collationList.push_back("");
m_collationList.sort();
// Editing an existing table?
if(m_bNewTable == false)
{
// Existing table, so load and set the current layout
m_table = *pdb.getTableByName(curTable);
ui->labelEditWarning->setVisible(!m_table.fullyParsed());
// Initialise the list of tracked columns for table layout changes
for(const auto& field : m_table.fields)
trackColumns[QString::fromStdString(field.name())] = QString::fromStdString(field.name());
// Set without rowid checkbox and schema dropdown. No need to trigger any events here as we're only loading a table exactly as it is stored by SQLite, so no need
// for error checking etc.
ui->checkWithoutRowid->blockSignals(true);
ui->checkWithoutRowid->setChecked(m_table.withoutRowidTable());
ui->checkWithoutRowid->blockSignals(false);
ui->checkStrict->blockSignals(true);
ui->checkStrict->setChecked(m_table.isStrict());
ui->checkStrict->blockSignals(false);
ui->comboSchema->blockSignals(true);
for(const auto& n : pdb.schemata) // Load list of database schemata
ui->comboSchema->addItem(QString::fromStdString(n.first));
ui->comboSchema->setCurrentText(QString::fromStdString(curTable.schema()));
ui->comboSchema->blockSignals(false);
if(m_table.primaryKey())
{
ui->checkWithoutRowid->blockSignals(true);
ui->comboOnConflict->setCurrentText(QString::fromStdString(m_table.primaryKey()->conflictAction()).toUpper());
ui->checkWithoutRowid->blockSignals(false);
}
populateFields();
populateIndexConstraints();
populateForeignKeys();
populateCheckConstraints();
} else {
for(const auto& n : pdb.schemata) // Load list of database schemata
ui->comboSchema->addItem(QString::fromStdString(n.first));
ui->comboSchema->setCurrentText("main"); // Always create tables in the main schema by default
ui->labelEditWarning->setVisible(false);
}
// Enable/disable remove constraint button depending on whether a constraint is selected
connect(ui->tableIndexConstraints, &QTableWidget::itemSelectionChanged, this, [this]() {
bool hasSelection = ui->tableIndexConstraints->selectionModel()->hasSelection();
ui->buttonRemoveIndexConstraint->setEnabled(hasSelection);
});
connect(ui->tableForeignKeys, &QTableWidget::itemSelectionChanged, this, [this]() {
bool hasSelection = ui->tableForeignKeys->selectionModel()->hasSelection();
ui->buttonRemoveForeignKey->setEnabled(hasSelection);
});
connect(ui->tableCheckConstraints, &QTableWidget::itemSelectionChanged, this, [this]() {
bool hasSelection = ui->tableCheckConstraints->selectionModel()->hasSelection();
ui->buttonRemoveCheckConstraint->setEnabled(hasSelection);
});
// And create a savepoint
pdb.setSavepoint(m_sRestorePointName);
// Update UI
ui->editTableName->setText(QString::fromStdString(curTable.name()));
updateColumnWidth();
// Allow editing of constraint columns by double clicking the columns column of the constraints table
connect(ui->tableIndexConstraints, &QTableWidget::itemDoubleClicked, this, &EditTableDialog::indexConstraintItemDoubleClicked);
connect(ui->tableForeignKeys, &QTableWidget::itemDoubleClicked, this, &EditTableDialog::foreignKeyItemDoubleClicked);
// (De-)activate fields
checkInput();
ui->sqlTextEdit->setReadOnly(true);
}
EditTableDialog::~EditTableDialog()
{
delete ui;
}
void EditTableDialog::keyPressEvent(QKeyEvent *evt)
{
if((evt->modifiers() & Qt::ControlModifier)
&& (evt->key() == Qt::Key_Enter || evt->key() == Qt::Key_Return))
{
accept();
return;
}
if(evt->key() == Qt::Key_Enter || evt->key() == Qt::Key_Return)
return;
QDialog::keyPressEvent(evt);
}
void EditTableDialog::updateColumnWidth()
{
ui->treeWidget->setColumnWidth(kName, 190);
ui->treeWidget->setColumnWidth(kType, 150);
ui->treeWidget->setColumnWidth(kNotNull, 25);
ui->treeWidget->setColumnWidth(kPrimaryKey, 25);
ui->treeWidget->setColumnWidth(kAutoIncrement, 25);
ui->treeWidget->setColumnWidth(kUnique, 25);
ui->treeWidget->setColumnWidth(kForeignKey, 500);
ui->tableIndexConstraints->setColumnWidth(kConstraintColumns, 180);
ui->tableIndexConstraints->setColumnWidth(kConstraintType, 130);
ui->tableIndexConstraints->setColumnWidth(kConstraintName, 120);
ui->tableIndexConstraints->setColumnWidth(kConstraintSql, 300);
ui->tableForeignKeys->setColumnWidth(kForeignKeyColumns, 120);
ui->tableForeignKeys->setColumnWidth(kForeignKeyName, 120);
ui->tableForeignKeys->setColumnWidth(kForeignKeyReferences, 300);
ui->tableForeignKeys->setColumnWidth(kForeignKeySql, 300);
ui->tableCheckConstraints->setColumnWidth(kCheckConstraintCheck, 400);
ui->tableCheckConstraints->setColumnWidth(kCheckConstraintName, 120);
ui->tableCheckConstraints->setColumnWidth(kCheckConstraintSql, 300);
}
void EditTableDialog::populateFields()
{
// Disable the itemChanged signal or the table item will be updated while filling the treewidget
ui->treeWidget->blockSignals(true);
ui->treeWidget->clear();
const auto& fields = m_table.fields;
const auto pk = m_table.primaryKey();
const auto pkColumns = m_table.primaryKeyColumns();
for(const sqlb::Field& f : fields)
{
QTreeWidgetItem *tbitem = new QTreeWidgetItem(ui->treeWidget);
tbitem->setFlags(tbitem->flags() | Qt::ItemIsEditable);
tbitem->setText(kName, QString::fromStdString(f.name()));
QComboBox* typeBox = new QComboBox(ui->treeWidget);
typeBox->setProperty("column", QString::fromStdString(f.name()));
typeBox->setEditable(!m_table.isStrict()); // Strict tables do not allow arbitrary types
typeBox->addItems(m_table.isStrict() ? DBBrowserDB::DatatypesStrict : DBBrowserDB::Datatypes);
int index = typeBox->findText(QString::fromStdString(f.type()), Qt::MatchFixedString);
if(index == -1)
{
// non standard named type
typeBox->addItem(QString::fromStdString(f.type()));
index = typeBox->count() - 1;
}
typeBox->setCurrentIndex(index);
typeBox->installEventFilter(this);
connect(typeBox, SIGNAL(currentIndexChanged(int)), this, SLOT(updateTypeAndCollation()));
ui->treeWidget->setItemWidget(tbitem, kType, typeBox);
tbitem->setCheckState(kNotNull, f.notnull() ? Qt::Checked : Qt::Unchecked);
tbitem->setCheckState(kPrimaryKey, pk && contains(pkColumns, f.name()) ? Qt::Checked : Qt::Unchecked);
tbitem->setCheckState(kAutoIncrement, pk && pk->autoIncrement() && contains(pkColumns, f.name()) ? Qt::Checked : Qt::Unchecked);
tbitem->setCheckState(kUnique, f.unique() ? Qt::Checked : Qt::Unchecked);
// For the default value check if it is surrounded by parentheses and if that's the case
// add a '=' character before the entire string to match the input format we're expecting
// from the user when using functions in the default value field.
if(!f.defaultValue().empty() && f.defaultValue().front() == '(' && f.defaultValue().back() == ')')
tbitem->setText(kDefault, "=" + QString::fromStdString(f.defaultValue()));
else
tbitem->setText(kDefault, QString::fromStdString(f.defaultValue()));
tbitem->setText(kCheck, QString::fromStdString(f.check()));
QComboBox* collationBox = new QComboBox(ui->treeWidget);
collationBox->setProperty("column", QString::fromStdString(f.name()));
collationBox->addItems(m_collationList);
index = collationBox->findText(QString::fromStdString(f.collation()), Qt::MatchCaseSensitive);
if(index == -1)
{
// some non-existing collation
collationBox->addItem(QString::fromStdString(f.collation()));
index = collationBox->count() - 1;
}
collationBox->setCurrentIndex(index);
collationBox->installEventFilter(this);
connect(collationBox, SIGNAL(currentIndexChanged(int)), this, SLOT(updateTypeAndCollation()));
ui->treeWidget->setItemWidget(tbitem, kCollation, collationBox);
auto fk = m_table.foreignKey({f.name()});
if(fk)
tbitem->setText(kForeignKey, QString::fromStdString(fk->toString()));
ui->treeWidget->addTopLevelItem(tbitem);
}
// and reconnect
ui->treeWidget->blockSignals(false);
}
void EditTableDialog::populateIndexConstraints()
{
// Disable the itemChanged signal or the table item will be updated while filling the treewidget
ui->tableIndexConstraints->blockSignals(true);
const auto& indexConstraints = m_table.indexConstraints();
ui->tableIndexConstraints->setRowCount(static_cast(indexConstraints.size()));
int row = 0;
for(const auto& it : indexConstraints)
{
// Columns
QTableWidgetItem* column = new QTableWidgetItem(QString::fromStdString(sqlb::joinStringVector(it.first, ",")));
column->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled);
column->setData(Qt::UserRole, QVariant::fromValue(it.first)); // Remember columns of constraint object. This is used for modifying it later
ui->tableIndexConstraints->setItem(row, kConstraintColumns, column);
// Type
QComboBox* type = new QComboBox(this);
type->addItem(tr("Primary Key"), "pk");
type->addItem(tr("Unique"), "u");
type->setCurrentIndex(it.second->isPrimaryKey() ? 0 : 1);
connect(type, static_cast(&QComboBox::currentIndexChanged), this, [this, type, it](int index) {
// Handle change of constraint type. Effectively this means removing the old constraint and replacing it by an entirely new one.
// Make sure there is only one primary key at a time
if(index == 0 && m_table.primaryKey())
{
QMessageBox::warning(this, qApp->applicationName(), tr("There can only be one primary key for each table. Please modify the existing primary "
"key instead."));
// Set combo box back to original constraint type
type->blockSignals(true);
type->setCurrentIndex(it.second->isPrimaryKey() ? 0 : 1);
type->blockSignals(false);
return;
}
// Remove old constraint
m_table.removeConstraint(it.second);
// Add new constraint depending on selected type
if(type->itemData(index).toString() == "pk")
{
auto pk = std::make_shared<:primarykeyconstraint>();
pk->setName(it.second->name());
m_table.addConstraint(it.first, pk);
} else if(type->itemData(index).toString() == "u") {
auto u = std::make_shared<:uniqueconstraint>();
u->setName(it.second->name());
m_table.addConstraint(it.first, u);
}
// Update SQL and view
populateFields();
populateIndexConstraints();
updateSqlText();
});
QTableWidgetItem* typeColumn = new QTableWidgetItem();
typeColumn->setData(Qt::UserRole, QVariant::fromValue<:shared_ptr>>(it.second)); // Remember address of constraint object. This is used for modifying it later
ui->tableIndexConstraints->setCellWidget(row, kConstraintType, type);
ui->tableIndexConstraints->setItem(row, kConstraintType, typeColumn);
// Name
QTableWidgetItem* name = new QTableWidgetItem(QString::fromStdString(it.second->name()));
name->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsEditable);
ui->tableIndexConstraints->setItem(row, kConstraintName, name);
// SQL
QTableWidgetItem* sql = new QTableWidgetItem(QString::fromStdString(it.second->toSql(it.first)));
sql->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled);
ui->tableIndexConstraints->setItem(row, kConstraintSql, sql);
row++;
}
ui->tableIndexConstraints->blockSignals(false);
}
void EditTableDialog::populateForeignKeys()
{
// Disable the itemChanged signal or the table item will be updated while filling the treewidget
ui->tableForeignKeys->blockSignals(true);
const auto& foreignKeys = m_table.foreignKeys();
ui->tableForeignKeys->setRowCount(static_cast(foreignKeys.size()));
int row = 0;
for(const auto& it : foreignKeys)
{
// Columns
QTableWidgetItem* columns = new QTableWidgetItem(QString::fromStdString(sqlb::joinStringVector(it.first, ",")));
columns->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsEditable);
columns->setData(Qt::UserRole, QVariant::fromValue(it.first)); // Remember columns of constraint object. This is used for modifying it later
ui->tableForeignKeys->setItem(row, kForeignKeyColumns, columns);
// Name
QTableWidgetItem* name = new QTableWidgetItem(QString::fromStdString(it.second->name()));
name->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsEditable);
ui->tableForeignKeys->setItem(row, kForeignKeyName, name);
// References
QTableWidgetItem* references = new QTableWidgetItem(QString::fromStdString(it.second->toString()));
references->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsEditable);
references->setData(Qt::UserRole, QVariant::fromValue(it.second)); // Remember address of constraint object. This is used for modifying it later
ui->tableForeignKeys->setItem(row, kForeignKeyReferences, references);
// SQL
QTableWidgetItem* sql = new QTableWidgetItem(QString::fromStdString(it.second->toSql(it.first)));
sql->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled);
ui->tableForeignKeys->setItem(row, kForeignKeySql, sql);
row++;
}
ui->tableForeignKeys->blockSignals(false);
}
void EditTableDialog::populateCheckConstraints()
{
// Disable the itemChanged signal or the table item will be updated while filling the treewidget
ui->tableCheckConstraints->blockSignals(true);
const auto& checkConstraints = m_table.checkConstraints();
ui->tableCheckConstraints->setRowCount(static_cast(checkConstraints.size()));
int row = 0;
for(const auto& it : checkConstraints)
{
// Check
QTableWidgetItem* check = new QTableWidgetItem(QString::fromStdString(it->expression()));
check->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsEditable);
check->setData(Qt::UserRole, QVariant::fromValue(it)); // Remember address of constraint object. This is used for modifying it later
ui->tableCheckConstraints->setItem(row, kCheckConstraintCheck, check);
// Name
QTableWidgetItem* name = new QTableWidgetItem(QString::fromStdString(it->name()));
name->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsEditable);
ui->tableCheckConstraints->setItem(row, kCheckConstraintName, name);
// SQL
QTableWidgetItem* sql = new QTableWidgetItem(QString::fromStdString(it->toSql()));
sql->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled);
ui->tableCheckConstraints->setItem(row, kCheckConstraintSql, sql);
row++;
}
ui->tableCheckConstraints->blockSignals(false);
}
void EditTableDialog::accept()
{
// Are we editing an already existing table or designing a new one? In the first case there is a table name set,
// in the latter the current table name is empty
if(m_bNewTable)
{
// Creation of new table
if(!pdb.executeSQL(m_table.sql(ui->comboSchema->currentText().toStdString())))
{
QMessageBox::warning(
this,
QApplication::applicationName(),
tr("Error creating table. Message from database engine:\n%1").arg(pdb.lastError()));
return;
}
} else {
// Editing of old table
// Apply all changes to the actual table in the database
if(!pdb.alterTable(curTable, m_table, trackColumns, ui->comboSchema->currentText().toStdString()))
{
QMessageBox::warning(this, QApplication::applicationName(), pdb.lastError());
return;
}
}
QDialog::accept();
}
void EditTableDialog::reject()
{
// Then rollback to our savepoint
pdb.revertToSavepoint(m_sRestorePointName);
QDialog::reject();
}
void EditTableDialog::updateSqlText()
{
ui->sqlTextEdit->setText(QString::fromStdString(m_table.sql(ui->comboSchema->currentText().toStdString())));
}
void EditTableDialog::checkInput()
{
std::string normTableName = ui->editTableName->text().toStdString();
bool valid = true;
if(normTableName.empty())
valid = false;
if(ui->treeWidget->topLevelItemCount() == 0)
valid = false;
if (normTableName != m_table.name()) {
const std::string oldTableName = m_table.name();
m_table.setName(normTableName);
// update fk's that refer to table itself recursively
const auto& fields = m_table.fields;
for(const sqlb::Field& f : fields) {
auto fk = m_table.foreignKey({f.name()});
if(fk && oldTableName == fk->table())
fk->setTable(normTableName);
}
populateFields();
}
updateSqlText();
ui->buttonBox->button(QDialogButtonBox::Ok)->setEnabled(valid);
}
void EditTableDialog::updateTypeAndCollation(QObject* object)
{
// Get sender combo box and retrieve field name from it
QComboBox* combo = qobject_cast(object);
if(!combo)
return;
QString column = combo->property("column").toString();
// Get type *and* collation combo box for this field
auto item = ui->treeWidget->findItems(column, Qt::MatchExactly, kName);
if(item.size() != 1)
return;
QComboBox* typeBox = qobject_cast(ui->treeWidget->itemWidget(item.front(), kType));
QComboBox* collationBox = qobject_cast(ui->treeWidget->itemWidget(item.front(), kCollation));
// Update table
if(typeBox && collationBox)
{
QString type = typeBox->currentText();
QString collation = collationBox->currentText();
for(size_t index=0; index < m_table.fields.size(); ++index)
{
if(m_table.fields.at(index).name() == column.toStdString())
{
m_table.fields.at(index).setType(type.toStdString());
m_table.fields.at(index).setCollation(collation.toStdString());
break;
}
}
checkInput();
}
}
void EditTableDialog::updateTypeAndCollation()
{
updateTypeAndCollation(sender());
}
bool EditTableDialog::eventFilter(QObject *object, QEvent *event)
{
if(event->type() == QEvent::FocusOut)
{
updateTypeAndCollation(object);
}
return false;
}
void EditTableDialog::fieldItemChanged(QTreeWidgetItem *item, int column)
{
size_t index = static_cast(ui->treeWidget->indexOfTopLevelItem(item));
if(index < m_table.fields.size())
{
sqlb::Field& field = m_table.fields.at(index);
QString oldFieldName = QString::fromStdString(field.name());
switch(column)
{
case kName:
{
// When a field of that name already exists, show a warning to the user and don't apply the new name. This is done by searching for an
// existing field with the new name. There is one exception, however, to this rule: if the field that we have found is the same field
// as the one we are currently trying to rename, this means the user is trying to rename the field to essentially the same name but
// with different case. Example: if I rename column 'COLUMN' to 'column', findField() is going to return the current field number
// because it's doing a case-independent search and it can't return another field number because SQLite prohibits duplicate field
// names (no matter the case). So when this happens we just allow the renaming because there's no harm to be expected from it.
auto foundField = sqlb::findField(m_table, item->text(column).toStdString());
if(foundField != m_table.fields.end() && foundField-m_table.fields.begin() != static_cast(index))
{
QMessageBox::warning(this, qApp->applicationName(), tr("There already is a field with that name. Please rename it first or choose a different "
"name for this field."));
// Reset the name to the old value but avoid calling this method again for that automatic change
ui->treeWidget->blockSignals(true);
item->setText(column, oldFieldName);
ui->treeWidget->blockSignals(false);
return;
}
// When editing an exiting table, check if any foreign keys would cause trouble in case this name is edited
if(!m_bNewTable)
{
const auto pk = m_table.primaryKey();
const auto pkColumns = m_table.primaryKeyColumns();
for(const auto& it : pdb.schemata[curTable.schema()].tables)
{
const sqlb::TablePtr& fkobj = it.second;
auto constraints = fkobj->foreignKeys();
for(const auto& fk : constraints)
{
if(fk.second->table() == m_table.name())
{
if(contains(fk.second->columns(), field.name()) || (pk && contains(pkColumns, field.name())))
{
QMessageBox::warning(this, qApp->applicationName(), tr("This column is referenced in a foreign key in table %1 and thus "
"its name cannot be changed.")
.arg(QString::fromStdString(fkobj->name())));
// Reset the name to the old value but avoid calling this method again for that automatic change
ui->treeWidget->blockSignals(true);
item->setText(column, oldFieldName);
ui->treeWidget->blockSignals(false);
return;
}
}
}
}
}
field.setName(item->text(column).toStdString());
m_table.renameKeyInAllConstraints(oldFieldName.toStdString(), item->text(column).toStdString());
qobject_cast(ui->treeWidget->itemWidget(item, kType))->setProperty("column", item->text(column));
qobject_cast(ui->treeWidget->itemWidget(item, kCollation))->setProperty("column", item->text(column));
// Update the field name in the map of old column names to new column names
if(!m_bNewTable)
{
for(const auto& it : trackColumns)
{
if(trackColumns[it.first] == oldFieldName)
trackColumns[it.first] = QString::fromStdString(field.name());
}
}
// Update the constraints view
populateIndexConstraints();
populateForeignKeys();
} break;
case kType:
case kCollation:
// see updateTypeAndCollation() SLOT
break;
case kPrimaryKey:
{
// Check if there already is a primary key
auto pk = m_table.primaryKey();
if(pk)
{
// There already is a primary key for this table. So edit that one as there always can only be one primary key anyway.
if(item->checkState(column) == Qt::Checked)
m_table.addKeyToConstraint(pk, field.name());
else
m_table.removeKeyFromConstraint(pk, field.name());
} else if(item->checkState(column) == Qt::Checked) {
// There is no primary key in the table yet. This means we need to add a default one.
m_table.addConstraint({sqlb::IndexedColumn(field.name(), false)}, std::make_shared<:primarykeyconstraint>());
}
if(item->checkState(column) == Qt::Checked)
{
// this will unset any other autoincrement
for(int i = 0; i < ui->treeWidget->topLevelItemCount(); ++i)
{
QTreeWidgetItem* tbitem = ui->treeWidget->topLevelItem(i);
if(tbitem != item)
tbitem->setCheckState(kAutoIncrement, Qt::Unchecked);
}
} else {
item->setCheckState(kAutoIncrement, Qt::Unchecked);
}
// Update the constraints view
populateIndexConstraints();
}
break;
case kNotNull:
{
// When editing an existing table and trying to set a column to Not Null an extra check is needed
if(!m_bNewTable && item->checkState(column) == Qt::Checked)
{
// Because our renameColumn() function fails when setting a column to Not Null when it already contains some NULL values
// we need to check for this case and cancel here. Maybe we can think of some way to modify the INSERT INTO ... SELECT statement
// to at least replace all troublesome NULL values by the default value
SqliteTableModel m(pdb, this);
m.setQuery(QString("SELECT COUNT(%1) FROM %2 WHERE coalesce(NULL,%3) IS NULL;").arg(
QString::fromStdString(sqlb::joinStringVector(sqlb::escapeIdentifier(pdb.getTableByName(curTable)->rowidColumns()), ",")),
QString::fromStdString(curTable.toString()),
QString::fromStdString(sqlb::escapeIdentifier(field.name()))));
if(!m.completeCache())
{
// If we couldn't load all data because the cancel button was clicked, just unset the checkbox again and stop.
item->setCheckState(column, Qt::Unchecked);
return;
}
if(m.data(m.index(0, 0)).toInt() > 0)
{
// There is a NULL value, so print an error message, uncheck the combobox, and return here
QMessageBox::information(this, qApp->applicationName(), tr("There is at least one row with this field set to NULL. "
"This makes it impossible to set this flag. Please change the table data first."));
item->setCheckState(column, Qt::Unchecked);
return;
}
}
field.setNotNull(item->checkState(column) == Qt::Checked);
}
break;
case kAutoIncrement:
{
bool ischecked = item->checkState(column) == Qt::Checked;
if(ischecked)
{
// First check if the contents of this column are all integers. If not this field cannot be set to AI
if(!m_bNewTable)
{
SqliteTableModel m(pdb, this);
m.setQuery(QString("SELECT COUNT(*) FROM %1 WHERE %2 <> CAST(%3 AS INTEGER);").arg(
QString::fromStdString(curTable.toString()),
QString::fromStdString(sqlb::escapeIdentifier(field.name())),
QString::fromStdString(sqlb::escapeIdentifier(field.name()))));
if(!m.completeCache())
{
// If we couldn't load all data because the cancel button was clicked, just unset the checkbox again and stop.
item->setCheckState(column, Qt::Unchecked);
return;
}
if(m.data(m.index(0, 0)).toInt() > 0)
{
// There is a non-integer value, so print an error message, uncheck the combobox, and return here
QMessageBox::information(this, qApp->applicationName(), tr("There is at least one row with a non-integer value in this field. "
"This makes it impossible to set the AI flag. Please change the table data first."));
item->setCheckState(column, Qt::Unchecked);
return;
}
}
// Make sure the data type is set to integer
QComboBox* comboType = qobject_cast(ui->treeWidget->itemWidget(item, kType));
comboType->setCurrentIndex(comboType->findText("INTEGER"));
item->setCheckState(kPrimaryKey, Qt::Checked);
// this will unset all other primary keys
// there can't be more than one autoincrement pk
for(int i = 0; i < ui->treeWidget->topLevelItemCount(); ++i)
{
QTreeWidgetItem* tbitem = ui->treeWidget->topLevelItem(i);
if(tbitem != item)
{
tbitem->setCheckState(kAutoIncrement, Qt::Unchecked);
tbitem->setCheckState(kPrimaryKey, Qt::Unchecked);
}
}
}
if(m_table.primaryKey())
m_table.primaryKey()->setAutoIncrement(ischecked);
}
break;
case kUnique:
{
// When editing an existing table and trying to set a column to unique an extra check is needed
if(!m_bNewTable && item->checkState(column) == Qt::Checked)
{
// Because our renameColumn() function fails when setting a column to unique when it already contains the same values
SqliteTableModel m(pdb, this);
m.setQuery(QString("SELECT COUNT(%2) FROM %1;").arg(
QString::fromStdString(curTable.toString()),
QString::fromStdString(sqlb::escapeIdentifier(field.name()))));
if(!m.completeCache())
{
// If we couldn't load all data because the cancel button was clicked, just unset the checkbox again and stop.
item->setCheckState(column, Qt::Unchecked);
return;
}
int rowcount = m.data(m.index(0, 0)).toInt();
m.setQuery(QString("SELECT COUNT(DISTINCT %2) FROM %1;").arg(
QString::fromStdString(curTable.toString()),
QString::fromStdString(sqlb::escapeIdentifier(field.name()))));
if(!m.completeCache())
{
// If we couldn't load all data because the cancel button was clicked, just unset the checkbox again and stop.
item->setCheckState(column, Qt::Unchecked);
return;
}
int uniquecount = m.data(m.index(0, 0)).toInt();
if(rowcount != uniquecount)
{
// There is a NULL value, so print an error message, uncheck the combobox, and return here
QMessageBox::information(this, qApp->applicationName(), tr("Column '%1' has duplicate data.\n").arg(QString::fromStdString(field.name()))
+ tr("This makes it impossible to enable the 'Unique' flag. Please remove the duplicate data, which will allow the 'Unique' flag to then be enabled."));
item->setCheckState(column, Qt::Unchecked);
return;
}
}
field.setUnique(item->checkState(column) == Qt::Checked);
}
break;
case kDefault:
{
QString new_value = item->text(column);
// If the default value isn't a SQL keyword perform an extra check: If it isn't numeric but doesn't start and end with quotes,
// add the quotes
if(new_value.size() && new_value.compare("null", Qt::CaseInsensitive) &&
new_value.compare("current_time", Qt::CaseInsensitive) &&
new_value.compare("current_date", Qt::CaseInsensitive) &&
new_value.compare("current_timestamp", Qt::CaseInsensitive))
{
QChar first_char = new_value.trimmed().at(0);
if(!((first_char == '\'' || first_char == '"') && new_value.trimmed().endsWith(first_char)))
{
bool is_numeric;
new_value.toDouble(&is_numeric);
if(!is_numeric)
{
if(new_value.trimmed().startsWith("=(") && new_value.trimmed().endsWith(')'))
{
new_value = new_value.trimmed().mid(1); // Leave the brackets as they are needed for a valid SQL expression
} else {
// Finally, the same check as in populateFields(),
// add "=" to GUI field, if a function surrounded by parentheses,
// or add quotes to both GUI field and stored SQL expression, otherwise.
if(new_value.front() == '(' && new_value.back() == ')') {
item->setText(column, '=' + new_value);
} else {
new_value = sqlb::escapeString(new_value);
item->setText(column, new_value);
}
}
}
}
}
field.setDefaultValue(new_value.toStdString());
}
break;
case kCheck:
field.setCheck(item->text(column).toStdString());
break;
case kForeignKey:
// handled in delegate
break;
}
}
checkInput();
}
void EditTableDialog::indexConstraintItemChanged(QTableWidgetItem* item)
{
// Find modified constraint
auto constraint = ui->tableIndexConstraints->item(item->row(), kConstraintType)->data(Qt::UserRole).value<:shared_ptr>>();
// Which column has been modified?
switch(item->column())
{
case kConstraintName:
constraint->setName(item->text().toStdString());
break;
}
// Update SQL
populateIndexConstraints();
checkInput();
}
void EditTableDialog::foreignKeyItemChanged(QTableWidgetItem* item)
{
// Find modified foreign key
auto fk = ui->tableForeignKeys->item(item->row(), kForeignKeyReferences)->data(Qt::UserRole).value<:shared_ptr>>();
// Which column has been modified?
switch(item->column())
{
case kForeignKeyName:
fk->setName(item->text().toStdString());
break;
}
// Update SQL
populateFields();
populateForeignKeys();
checkInput();
}
void EditTableDialog::checkConstraintItemChanged(QTableWidgetItem* item)
{
// Find modified check constraint
auto c = ui->tableCheckConstraints->item(item->row(), kCheckConstraintCheck)->data(Qt::UserRole).value<:shared_ptr>>();
// Which column has been modified?
switch(item->column())
{
case kCheckConstraintCheck:
c->setExpression(item->text().toStdString());
break;
case kCheckConstraintName:
c->setName(item->text().toStdString());
break;
}
// Update SQL
populateFields();
populateCheckConstraints();
checkInput();
}
void EditTableDialog::indexConstraintItemDoubleClicked(QTableWidgetItem* item)
{
// Check whether the double clicked item is in the columns column
if(item->column() == kConstraintColumns)
{
sqlb::IndexedColumnVector indexed_columns = ui->tableIndexConstraints->item(item->row(), item->column())->data(Qt::UserRole).value<:indexedcolumnvector>();
sqlb::StringVector columns;
std::transform(indexed_columns.begin(), indexed_columns.end(), std::back_inserter(columns), [](const auto& e) { return e.name(); });
// Show the select items popup dialog
SelectItemsPopup* dialog = new SelectItemsPopup(m_table.fieldNames(), columns, this);
QRect item_rect = ui->tableIndexConstraints->visualItemRect(item);
dialog->move(ui->tableIndexConstraints->mapToGlobal(QPoint(ui->tableIndexConstraints->x() + item_rect.x(),
ui->tableIndexConstraints->y() + item_rect.y() + item_rect.height() / 2)));
dialog->show();
// Get constraint
auto constraint = ui->tableIndexConstraints->item(item->row(), kConstraintType)->data(Qt::UserRole).value<:shared_ptr>>();
// When clicking the Apply button in the popup dialog, save the new columns list
connect(dialog, &SelectItemsPopup::accepted, this, [this, dialog, constraint, columns]() {
// Check if column selection changed at all
sqlb::StringVector new_columns = dialog->selectedItems();
if(columns != new_columns)
{
// Remove the constraint with the old columns and add a new one with the new columns
m_table.removeConstraint(constraint);
if(constraint->isPrimaryKey())
m_table.addConstraint(new_columns, std::dynamic_pointer_cast<:primarykeyconstraint>(constraint));
else
m_table.addConstraint(new_columns, std::dynamic_pointer_cast<:uniqueconstraint>(constraint));
// Update the UI
populateFields();
populateIndexConstraints();
updateSqlText();
}
});
}
}
void EditTableDialog::foreignKeyItemDoubleClicked(QTableWidgetItem* item)
{
// Check whether the double clicked item is in the columns column
if(item->column() == kForeignKeyColumns)
{
auto columns = ui->tableForeignKeys->item(item->row(), item->column())->data(Qt::UserRole).value<:stringvector>();
auto fk = item->tableWidget()->item(item->row(), kForeignKeyReferences)->data(Qt::UserRole).value<:shared_ptr>>();
// Show the select items popup dialog
SelectItemsPopup* dialog = new SelectItemsPopup(m_table.fieldNames(), columns, this);
QRect item_rect = ui->tableForeignKeys->visualItemRect(item);
dialog->move(item->tableWidget()->mapToGlobal(QPoint(ui->tableForeignKeys->x() + item_rect.x(),
ui->tableForeignKeys->y() + item_rect.y() + item_rect.height() / 2)));
dialog->show();
// When clicking the Apply button in the popup dialog, save the new columns list
connect(dialog, &SelectItemsPopup::accepted, this, [this, dialog, fk, columns]() {
// Check if column selection changed at all
sqlb::StringVector new_columns = dialog->selectedItems();
if(columns != new_columns)
{
// Remove the constraint with the old columns and add a new one with the new columns
m_table.removeConstraint(fk);
m_table.addConstraint(new_columns, fk);
// Update the UI
populateFields();
populateForeignKeys();
updateSqlText();
}
});
}
}
void EditTableDialog::addField()
{
QTreeWidgetItem *tbitem = new QTreeWidgetItem(ui->treeWidget);
tbitem->setFlags(tbitem->flags() | Qt::ItemIsEditable);
// Find an unused name for the field by starting with 'Fieldx' where x is the number of fields + 1.
// If this name happens to exist already, increase x by one until we find an unused name.
{
int field_number = ui->treeWidget->topLevelItemCount();
std::string field_name;
do
{
field_name = "Field" + std::to_string(field_number);
field_number++;
} while(sqlb::findField(m_table, field_name) != m_table.fields.end());
tbitem->setText(kName, QString::fromStdString(field_name));
}
QComboBox* typeBox = new QComboBox(ui->treeWidget);
typeBox->setProperty("column", tbitem->text(kName));
typeBox->setEditable(!m_table.isStrict()); // Strict tables do not allow arbitrary types
typeBox->addItems(m_table.isStrict() ? DBBrowserDB::DatatypesStrict : DBBrowserDB::Datatypes);
int defaultFieldTypeIndex = Settings::getValue("db", "defaultfieldtype").toInt();
if (defaultFieldTypeIndex < DBBrowserDB::Datatypes.count())
{
typeBox->setCurrentIndex(defaultFieldTypeIndex);
}
ui->treeWidget->setItemWidget(tbitem, kType, typeBox);
typeBox->installEventFilter(this);
connect(typeBox, SIGNAL(currentIndexChanged(int)), this, SLOT(updateTypeAndCollation()));
QComboBox* collationBox = new QComboBox(ui->treeWidget);
collationBox->setProperty("column", tbitem->text(kName));
collationBox->addItems(m_collationList);
collationBox->setCurrentIndex(collationBox->findText(""));
ui->treeWidget->setItemWidget(tbitem, kCollation, collationBox);
collationBox->installEventFilter(this);
connect(collationBox, SIGNAL(currentIndexChanged(int)), this, SLOT(updateTypeAndCollation()));
tbitem->setCheckState(kNotNull, Qt::Unchecked);
tbitem->setCheckState(kPrimaryKey, Qt::Unchecked);
tbitem->setCheckState(kAutoIncrement, Qt::Unchecked);
tbitem->setCheckState(kUnique, Qt::Unchecked);
ui->treeWidget->addTopLevelItem(tbitem);
ui->treeWidget->scrollToBottom();
ui->treeWidget->editItem(tbitem, 0);
// add field to table object
m_table.fields.emplace_back(tbitem->text(kName).toStdString(), typeBox->currentText().toStdString());
// Add the new column to the list of tracked columns to indicate it has been added
if(!m_bNewTable)
trackColumns.insert({QString(), tbitem->text(kName)});
checkInput();
}
void EditTableDialog::removeField()
{
// Is there any item selected to delete?
if(!ui->treeWidget->currentItem())
return;
// If we are editing an existing table, ask the user for confirmation
if(!m_bNewTable)
{
QString msg = tr("Are you sure you want to delete the field '%1'?\nAll data currently stored in this field will be lost.").arg(ui->treeWidget->currentItem()->text(0));
if(QMessageBox::warning(this, QApplication::applicationName(), msg, QMessageBox::Yes | QMessageBox::No, QMessageBox::No) == QMessageBox::No)
return;
// Update the map of tracked columns to indicate the column is deleted
QString name = ui->treeWidget->currentItem()->text(0);
for(const auto& it : trackColumns)
{
if(trackColumns[it.first] == name)
trackColumns[it.first] = QString();
}
}
// Just delete that item. At this point there is no DB table to edit or data to be lost anyway
m_table.fields.erase(m_table.fields.begin() + ui->treeWidget->indexOfTopLevelItem(ui->treeWidget->currentItem()));
m_table.removeKeyFromAllConstraints(ui->treeWidget->currentItem()->text(kName).toStdString());
delete ui->treeWidget->currentItem();
// Update the constraints view
populateIndexConstraints();
populateForeignKeys();
checkInput();
}
void EditTableDialog::fieldSelectionChanged()
{
bool hasSelection = ui->treeWidget->selectionModel()->hasSelection();
// Enable the remove and the move up/down buttons if a field is selected, disable it otherwise
ui->removeFieldButton->setEnabled(hasSelection);
ui->buttonMoveUp->setEnabled(hasSelection);
ui->buttonMoveDown->setEnabled(hasSelection);
// If the selected line is the first one disable the move up button, it it's the last one disable the move down button
if(hasSelection)
{
ui->buttonMoveUp->setEnabled(ui->treeWidget->selectionModel()->currentIndex().row() != 0);
ui->buttonMoveTop->setEnabled(ui->buttonMoveUp->isEnabled());
ui->buttonMoveDown->setEnabled(ui->treeWidget->selectionModel()->currentIndex().row() != ui->treeWidget->topLevelItemCount() - 1);
ui->buttonMoveBottom->setEnabled(ui->buttonMoveDown->isEnabled());
}
}
void EditTableDialog::moveUp()
{
moveCurrentField(MoveUp);
}
void EditTableDialog::moveDown()
{
moveCurrentField(MoveDown);
}
void EditTableDialog::moveTop()
{
moveCurrentField(MoveTop);
}
void EditTableDialog::moveBottom()
{
moveCurrentField(MoveBottom);
}
void EditTableDialog::moveCurrentField(MoveFieldDirection dir)
{
int currentRow = ui->treeWidget->currentIndex().row();
int currentCol = ui->treeWidget->currentIndex().column();
int newRow;
if(dir == MoveUp)
newRow = currentRow - 1;
else if(dir == MoveDown)
newRow = currentRow + 1;
else if(dir == MoveTop)
newRow = 0;
else if(dir == MoveBottom)
newRow = ui->treeWidget->topLevelItemCount() - 1;
else
return;
// Save the comboboxes first by making copies
QComboBox* newCombo[2];
for(int c=0;c<2;c++)
{
int column = (c == 0 ? kType : kCollation);
QComboBox* oldCombo = qobject_cast(ui->treeWidget->itemWidget(ui->treeWidget->topLevelItem(currentRow), column));
newCombo[c] = new QComboBox(ui->treeWidget);
newCombo[c]->setProperty("column", oldCombo->property("column"));
newCombo[c]->installEventFilter(this);
connect(newCombo[c], SIGNAL(currentIndexChanged(int)), this, SLOT(updateTypeAndCollation()));
newCombo[c]->setEditable(oldCombo->isEditable());
for(int i=0; i < oldCombo->count(); ++i)
newCombo[c]->addItem(oldCombo->itemText(i));
newCombo[c]->setCurrentIndex(oldCombo->currentIndex());
}
// Now, just remove the item and insert it at it's new position, then restore the combobox
QTreeWidgetItem* item = ui->treeWidget->takeTopLevelItem(currentRow);
ui->treeWidget->insertTopLevelItem(newRow, item);
ui->treeWidget->setItemWidget(item, kType, newCombo[0]);
ui->treeWidget->setItemWidget(item, kCollation, newCombo[1]);
// Select the old item at its new position
ui->treeWidget->setCurrentIndex(ui->treeWidget->currentIndex().sibling(newRow, currentCol));
// Finally update the table SQL
sqlb::Field temp = m_table.fields[static_cast(currentRow)];
m_table.fields.erase(m_table.fields.begin() + currentRow);
m_table.fields.insert(m_table.fields.begin() + newRow, temp);
// Update the SQL preview
updateSqlText();
}
void EditTableDialog::setWithoutRowid(bool without_rowid)
{
if(without_rowid)
{
// Before setting the without rowid flag, first perform a check to see if the table meets all the required criteria for without rowid tables
const auto pk = m_table.primaryKey();
if(!pk || pk->autoIncrement())
{
QMessageBox::information(this, QApplication::applicationName(),
tr("Please add a field which meets the following criteria before setting the without rowid flag:\n"
" - Primary key flag set\n"
" - Auto increment disabled"));
// Reset checkbox state to unchecked. Block any signals while doing this in order to avoid an extra call to
// this function being triggered.
ui->checkWithoutRowid->blockSignals(true);
ui->checkWithoutRowid->setChecked(false);
ui->checkWithoutRowid->blockSignals(false);
return;
}
// If it does, set the without rowid flag of the table
m_table.setWithoutRowidTable(true);
} else {
// If the without rowid flag is unset no further checks are required. Just unset the without rowid flag
m_table.setWithoutRowidTable(false);
}
// Update the SQL preview
updateSqlText();
}
void EditTableDialog::setStrict(bool strict)
{
// Set the strict option
m_table.setStrict(strict);
// Replace list of possible data types in each type combo box
for(int i=0;itreeWidget->topLevelItemCount();i++)
{
QComboBox* w = qobject_cast(ui->treeWidget->itemWidget(ui->treeWidget->topLevelItem(i), kType));
QString value = w->currentText();
w->blockSignals(true);
w->clear();
w->addItems(strict ? DBBrowserDB::DatatypesStrict : DBBrowserDB::Datatypes);
w->setEditable(!strict); // Strict tables do not allow arbitrary types
int pos = w->findText(value, Qt::MatchFixedString);
w->blockSignals(false);
if(pos >= 0)
{
w->setCurrentIndex(pos);
} else {
// When the old value cannot be found we default to ANY for strict tables. For ordinary tables we just add the former value.
if(strict)
w->setCurrentText("ANY");
else
w->setCurrentText(value);
}
}
// Update the SQL preview
updateSqlText();
}
void EditTableDialog::changeSchema(int /*schema_idx*/)
{
// Update the SQL preview
updateSqlText();
}
void EditTableDialog::setOnConflict(int on_conflict_idx)
{
const QString &on_conflict = ui->comboOnConflict->itemText(on_conflict_idx);
if(m_table.primaryKey())
{
m_table.primaryKey()->setConflictAction(on_conflict.toStdString());
} else {
QMessageBox::information(this, QApplication::applicationName(),
tr("Please add a field which meets the following criteria before setting the on conflict action:\n"
" - Primary key flag set"));
ui->comboOnConflict->blockSignals(true);
ui->comboOnConflict->setCurrentText(QString());
ui->comboOnConflict->blockSignals(false);
return;
}
// Update the SQL preview
updateSqlText();
}
void EditTableDialog::removeIndexConstraint()
{
// Is there any item selected to delete?
if(!ui->tableIndexConstraints->currentItem())
return;
// Find constraint to delete
int row = ui->tableIndexConstraints->currentRow();
auto constraint = ui->tableIndexConstraints->item(row, kConstraintType)->data(Qt::UserRole).value<:shared_ptr>>();
// Remove the constraint. If there is more than one constraint with this combination of columns and constraint type, only delete the first one.
m_table.removeConstraint(constraint);
ui->tableIndexConstraints->removeRow(ui->tableIndexConstraints->currentRow());
// Update SQL and view
updateSqlText();
populateFields();
}
void EditTableDialog::addIndexConstraint(bool primary_key)
{
// There can only be one primary key
if(primary_key)
{
if(m_table.primaryKey())
{
QMessageBox::information(this, qApp->applicationName(), tr("There can only be one primary key for each table. Please modify the existing primary "
"key instead."));
return;
}
// Create new constraint
m_table.addConstraint(sqlb::IndexedColumnVector{}, std::make_shared<:primarykeyconstraint>());
} else
m_table.addConstraint(sqlb::IndexedColumnVector{}, std::make_shared<:uniqueconstraint>());
// Update SQL and view
populateFields();
populateIndexConstraints();
updateSqlText();
}
void EditTableDialog::addForeignKey()
{
m_table.addConstraint(sqlb::StringVector{}, std::make_shared<:foreignkeyclause>());
// Update SQL and view
populateFields();
populateForeignKeys();
updateSqlText();
}
void EditTableDialog::removeForeignKey()
{
// Is there any item selected to delete?
if(!ui->tableForeignKeys->currentItem())
return;
// Find and remove the selected foreign key
int row = ui->tableForeignKeys->currentRow();
auto fk = ui->tableForeignKeys->item(row, kForeignKeyReferences)->data(Qt::UserRole).value<:shared_ptr>>();
m_table.removeConstraint(fk);
ui->tableForeignKeys->removeRow(row);
// Update SQL and view
updateSqlText();
populateFields();
}
void EditTableDialog::addCheckConstraint()
{
std::string check_name;
// Find an unused name for the check by starting with 'Checkx' where x is the number of checks + 1.
// If this name happens to exist already, increase x by one until we find an unused name.
{
int check_number = ui->tableCheckConstraints->rowCount() + 1;
do
{
check_name = "Check" + std::to_string(check_number);
check_number++;
} while(!ui->tableCheckConstraints->findItems(QString::fromStdString(check_name), Qt::MatchExactly).isEmpty());
}
auto constraint = std::make_shared<:checkconstraint>();
constraint->setName(check_name);
m_table.addConstraint(constraint);
// Update SQL and view
populateCheckConstraints();
updateSqlText();
}
void EditTableDialog::removeCheckConstraint()
{
// Is there any item selected to delete?
if(!ui->tableCheckConstraints->currentItem())
return;
// Find and remove the selected check constraint
int row = ui->tableCheckConstraints->currentRow();
auto check = ui->tableCheckConstraints->item(row, kCheckConstraintCheck)->data(Qt::UserRole).value<:shared_ptr>>();
m_table.removeConstraint(check);
ui->tableCheckConstraints->removeRow(row);
// Update SQL
updateSqlText();
}