The UI Class: MainWindow
The example's UI is a conventional Qt GUI application inheriting QMainWindow and the class generated by Qt Designer:
class MainWindow : public QMainWindow,
private Ui::SchemaMainWindow
{
Q_OBJECT
public:
MainWindow();
private Q_SLOTS:
void schemaSelected(int index);
void instanceSelected(int index);
void validate();
void textChanged();
private:
void moveCursor(int line, int column);
};
The constructor fills the schema and instance QComboBox selections with the predefined schemas and instances and connects their currentIndexChanged() signals to the window's schemaSelected() resp. instanceSelected() slot. Furthermore the signal-slot connections for the validation button and the instance editor are set up.
The call to schemaSelected(0) and instanceSelected(0) will trigger the validation of the initial Contact Schema example.
MainWindow::MainWindow()
{
setupUi(this);
new XmlSyntaxHighlighter(schemaView->document());
new XmlSyntaxHighlighter(instanceEdit->document());
schemaSelection->addItem(tr("Contact Schema"));
schemaSelection->addItem(tr("Recipe Schema"));
schemaSelection->addItem(tr("Order Schema"));
instanceSelection->addItem(tr("Valid Contact Instance"));
instanceSelection->addItem(tr("Invalid Contact Instance"));
connect(schemaSelection, SIGNAL(currentIndexChanged(int)), SLOT(schemaSelected(int)));
connect(instanceSelection, SIGNAL(currentIndexChanged(int)), SLOT(instanceSelected(int)));
connect(validateButton, SIGNAL(clicked()), SLOT(validate()));
connect(instanceEdit, SIGNAL(textChanged()), SLOT(textChanged()));
validationStatus->setAlignment(Qt::AlignCenter | Qt::AlignVCenter);
schemaSelected(0);
instanceSelected(0);
}
In the schemaSelected() slot the content of the instance selection is adapted to the selected schema and the corresponding schema is loaded from the resource file and displayed in the schema viewer. At the end of the method a revalidation is triggered.
void MainWindow::schemaSelected(int index)
{
instanceSelection->clear();
if (index == 0) {
instanceSelection->addItem(tr("Valid Contact Instance"));
instanceSelection->addItem(tr("Invalid Contact Instance"));
} else if (index == 1) {
instanceSelection->addItem(tr("Valid Recipe Instance"));
instanceSelection->addItem(tr("Invalid Recipe Instance"));
} else if (index == 2) {
instanceSelection->addItem(tr("Valid Order Instance"));
instanceSelection->addItem(tr("Invalid Order Instance"));
}
textChanged();
QFile schemaFile(QString(":/schema_%1.xsd").arg(index));
schemaFile.open(QIODevice::ReadOnly);
const QString schemaText(QString::fromUtf8(schemaFile.readAll()));
schemaView->setPlainText(schemaText);
validate();
}
In the instanceSelected() slot the selected instance is loaded from the resource file and loaded into the instance editor an the revalidation is triggered again.
void MainWindow::instanceSelected(int index)
{
QFile instanceFile(QString(":/instance_%1.xml").arg((2*schemaSelection->currentIndex()) + index));
instanceFile.open(QIODevice::ReadOnly);
const QString instanceText(QString::fromUtf8(instanceFile.readAll()));
instanceEdit->setPlainText(instanceText);
validate();
}
The validate() slot does the actual work in this example. At first it stores the content of the schema viewer and the editor into temporary variables. Then it instanciates a MessageHandler object which inherits from QAbstractMessageHandler and is a convenience class to store error messages from the XmlPatterns system.
class MessageHandler : public QAbstractMessageHandler
{
public:
MessageHandler()
: QAbstractMessageHandler(0)
{
}
QString statusMessage() const
{
return m_description;
}
int line() const
{
return m_sourceLocation.line();
}
int column() const
{
return m_sourceLocation.column();
}
protected:
virtual void handleMessage(QtMsgType type, const QString &description,
const QUrl &identifier, const QSourceLocation &sourceLocation)
{
Q_UNUSED(type);
Q_UNUSED(identifier);
m_messageType = type;
m_description = description;
m_sourceLocation = sourceLocation;
}
private:
QtMsgType m_messageType;
QString m_description;
QSourceLocation m_sourceLocation;
};
After the QXmlSchema is instanciated and the message handler set on it, the load() method is called with the schema data as argument. If the schema is invalid or a parsing error has occured, isValid() returns false and the error is flagged in errorOccurred. If the loading was successful, a QXmlSchemaValidator is instanciated and the schema passed in the constructor. A call to validate() will validate the passed XML instance data against the XML schema. The return value of that method signals whether the validation was successful. Depending on the success the status label is set to 'validation successful' or the error message stored in the MessageHandler
The rest of the code does only some fancy coloring and eyecandy.
void MainWindow::validate()
{
const QByteArray schemaData = schemaView->toPlainText().toUtf8();
const QByteArray instanceData = instanceEdit->toPlainText().toUtf8();
MessageHandler messageHandler;
QXmlSchema schema;
schema.setMessageHandler(&messageHandler);
schema.load(schemaData);
bool errorOccurred = false;
if (!schema.isValid()) {
errorOccurred = true;
} else {
QXmlSchemaValidator validator(schema);
if (!validator.validate(instanceData))
errorOccurred = true;
}
if (errorOccurred) {
validationStatus->setText(messageHandler.statusMessage());
moveCursor(messageHandler.line(), messageHandler.column());
} else {
validationStatus->setText(tr("validation successful"));
}
const QString styleSheet = QString("QLabel {background: %1; padding: 3px}")
.arg(errorOccurred ? QColor(Qt::red).lighter(160).name() :
QColor(Qt::green).lighter(160).name());
validationStatus->setStyleSheet(styleSheet);
}