4

我必须实现一个 xml 解析器,该解析器在运行时构造 Qt 对象并将其放置在小部件中。我基于这个写了一个解析器。一个模块表示QWidget将保存QPushButtonQCheckBox实例。xml是 a QXmlStreamReader,就像 Qt 参考中的示例一样。

所以,在这里我尝试解析这个非常基本的 xml 文件:

<somexml version="1.0">
    <module>
        <button label="Send UART" define="BUTTON0" pos="30, 30" size="20, 10">
            <action cmd="sendCom" data="0xAA 0xBB 0xCC"/>
        </button>
        <checkbox label="Send UART" define="CHECKBOX0" pos="250, 30" size="20, 10">
            <action cmd="sendCom" data="test"/>
        </checkbox>
        <button label="Check BIT" define="BUTTON1" pos="140, 30" size="20, 10">
            <action cmd="setDef" data="BUTTON0=1"/>
        </button>   
    </module>
</somexml>

一个模块元素将创建一个小部件来保存标签内的复选框和按钮。所以在这里:

void XmlReader::readXml()
{
    while (xml.readNextStartElement())
    {
        if (xml.name() == "module")
        {
            readModule();
        }
    }
}

我希望阅读模块元素并输入它的范围readModule。在里面readModule我真的很难遵循解析流程。

这是实现:

void XmlReader::readModule()
{
    Q_ASSERT(xml.isStartElement() && xml.name() == "module");


    while (xml.readNextStartElement())
    {
        if (xml.name() == "button")
        {
            readButton(widget);
        }
        else if (xml.name() == "checkbox")
        {
            readCheckBox(widget);
        }
        else
        {
            xml.skipCurrentElement();
        }
    }
}

它只会到达buttonxml 文件中的第一个元素并创建一个按钮。发送 UART。

当我将实现更改readXml为:

void XmlReader::readXml()
{
    while (xml.readNextStartElement())
    {
        if (xml.name() == "module")
        {
            readModule();
        }
        else if (xml.name() == "button")
        {
            readButton(widget);
        }
        else if (xml.name() == "checkbox")
        {
            readCheckBox(widget);
        }
        else
        { 
            xml.skipCurrentElement();
        }
    }
}

它将到达第一个按钮和第一个复选框。我错过了什么?

编辑:添加了 readButton 的实现

void XmlReader::readButton(Widget *widget)
{
    Q_ASSERT(xml.isStartElement() && xml.name() == "button");

    QString label = xml.attributes().value("label").toString();
    QString define = xml.attributes().value("define").toString();

    QString pos = xml.attributes().value("pos").toString();
    QStringList posList = pos.split(",");
    int posX = posList[0].toInt();
    int posY = posList[1].toInt();

    QString size = xml.attributes().value("size").toString();
    QStringList sizeList= size.split(",");
    int sizeX = sizeList[0].toInt();
    int sizeY = sizeList[1].toInt();

    QString cmd, data;

    while (xml.readNextStartElement())
    {
        if (xml.name() == "action") 
        {
            cmd = xml.attributes().value("cmd").toString();
            data = xml.attributes().value("data").toString();
        }
        else
        {
            xml.skipCurrentElement();
        }
    }

    if (cmd == "sendCom")
    {
        widget->createButton(label, define, posX, posY,
                SLOT(sendCom(QString)), data);
    }
    else if (cmd == "setDef")
    {
        widget->createButton(label, define, posX, posY,
                SLOT(setDef(QString)), data);
    }
}
4

1 回答 1

3

您需要使用xml.skipCurrentElement();afterreadButton(widget);readCheckBox(widget);。没有它xml.readNextStartElement()读取元素的结尾并打破循环。

void XmlReader::readModule()
{
    Q_ASSERT(xml.isStartElement() && xml.name() == "module");

    while (xml.readNextStartElement())
    {
        if (xml.name() == "button")
        {
            readButton(widget);
            xml.skipCurrentElement();
        }
        else if (xml.name() == "checkbox")
        {
            readCheckBox(widget);
            xml.skipCurrentElement();
        }
        else
        {
            xml.skipCurrentElement();
        }
    }
}
于 2016-01-26T19:40:44.157 回答