如何在我的 Qt 应用程序中简单地解析 CSS 之类的 (!) 文件?
How can I simply parse a CSS like (!) file in my Qt application?
我有一个类似 *.css(级联样式表)格式的文档,但它有自己的关键字。实际上它是一个个性化的 css(我称之为 *.pss),具有自己的标签和属性。在这里我有一段摘录:
/* CSS like style sheet file *.pss */
@include "otherStyleSheet.pss";
/* comment */
[propertyID="1230000"] {
fillColor : #f3f1ed;
minSize : 5;
lineWidth : 3;
}
/* sphere */
[propertyID="124???|123000"] {
lineType : dotted;
}
/* square */
[propertyID="125???"] {
lineType : thinline;
}
/* ring */
[propertyID="133???"] {
lineType : thickline;
[hasInnerRing=true] {
innerLineType : thinline;
}
}
我想很容易地解析它,是否已经有 Qt 的现成可用的东西?什么是最简单的方法?
由于 *.css 有自己的关键字,我对 CSS 解析器不感兴趣。
解析 *.pss 后我的进一步意图是将其属性存储在 Model 结构中。
Qt 中没有任何东西 public。您当然可以自由使用 Qt 的私有 CSS 解析器 - 您可以复制它并进行修改以满足您的需要。
参见 qtbase/src/gui/text/qcssparser_p.h
, in qtbase/src/gui/text
。
好消息是,对于上面显示的示例,修改非常小。 Qt 的 CSS 解析器已经支持 @import
,所以我们只有额外的一点语法,就是 nested selector syntax。如果没有该语法,您可以按原样使用 QCss::Parser
。解析器以灵活的方式编写,您无需担心正式的 CSS 关键字:它仍将允许您访问所有声明,无论它们从正式的 CSS 点是否有意义查看与否。
迭代解析树非常简单:
int main() {
QCss::Parser parser(pss);
QCss::StyleSheet styleSheet;
if (!parser.parse(&styleSheet))
return 1;
for (auto rule : styleSheet.styleRules) {
qDebug() << "** Rule **";
for (auto sel : rule.selectors) {
for (auto bSel : sel.basicSelectors)
qDebug() << bSel;
}
for (auto decl : rule.declarations)
qDebug() << decl;
}
}
输出是我们所期望的:
** Rule **
BasicSelector "propertyID"="1230000"
Declaration "fillColor" = '#f3f1ed' % QColor(ARGB 1, 0.952941, 0.945098, 0.929412)
Declaration "minSize" = '5' % 5
Declaration "lineWidth" = '3'
** Rule **
BasicSelector "propertyID"="124???|123000"
Declaration "lineType" = 'dotted'
** Rule **
BasicSelector "propertyID"="125???"
Declaration "lineType" = 'thinline'
** Rule **
BasicSelector "propertyID"="133???"
Declaration "lineType" = 'thickline'
我们必须自己为 QCss
类 实现调试流运算符:
QDebug operator<<(QDebug dbg, const QCss::AttributeSelector & sel) {
QDebugStateSaver saver(dbg);
dbg.noquote().nospace() << "\"" << sel.name << "\"";
switch (sel.valueMatchCriterium) {
case QCss::AttributeSelector::MatchEqual:
dbg << "="; break;
case QCss::AttributeSelector::MatchContains:
dbg << "~="; break;
case QCss::AttributeSelector::MatchBeginsWith:
dbg << "^="; break;
case QCss::AttributeSelector::NoMatch:
break;
}
if (sel.valueMatchCriterium != QCss::AttributeSelector::NoMatch && !sel.value.isEmpty())
dbg << "\"" << sel.value << "\"";
return dbg;
}
QDebug operator<<(QDebug dbg, const QCss::BasicSelector & sel) {
QDebugStateSaver saver(dbg);
dbg.noquote().nospace() << "BasicSelector";
if (!sel.elementName.isEmpty())
dbg << " #" << sel.elementName;
for (auto & id : sel.ids)
dbg << " id:" << id;
for (auto & aSel : sel.attributeSelectors)
dbg << " " << aSel;
return dbg;
}
在遍历声明时,QCss::parser
已经为我们解释了一些标准值,例如颜色、整数等
QDebug operator<<(QDebug dbg, const QCss::Declaration & decl) {
QDebugStateSaver saver(dbg);
dbg.noquote().nospace() << "Declaration";
dbg << " \"" << decl.d->property << "\" = ";
bool first = true;
for (auto value : decl.d->values) {
if (!first) dbg << ", ";
dbg << "\'" << value.toString() << "\'";
first = false;
}
if (decl.d->property == "fillColor")
dbg << " % " << decl.colorValue();
else if (decl.d->property == "minSize") {
int i;
if (decl.intValue(&i)) dbg << " % " << i;
}
return dbg;
}
最后,要解析的样板文件和样式表:
// https://github.com/KubaO/Whosebugn/tree/master/questions/css-like-parser-31583622
#include <QtGui>
#include <private/qcssparser_p.h>
const char pss[] =
"/* @include \"otherStyleSheet.pss\"; */ \
[propertyID=\"1230000\"] { \
fillColor : #f3f1ed; \
minSize : 5; \
lineWidth : 3; \
} \
\
/* sphere */ \
[propertyID=\"124???|123000\"] { \
lineType : dotted; \
} \
\
/* square */ \
[propertyID=\"125???\"] { \
lineType : thinline; \
} \
\
/* ring */ \
[propertyID=\"133???\"] { \
lineType : thickline; \
/*[hasInnerRing=true] { \
innerLineType : thinline; \
}*/ \
}";
对嵌套的支持selectors/rules可以通过修改解析器源来实现。使 Parser::parseRuleset
递归所需的更改非常小。我将把它留作 reader :)
的练习
总而言之,我认为重用现有的解析器比推出自己的解析器要容易得多,尤其是当您的用户不可避免地希望您支持越来越多的 CSS 规范时。
好吧,我猜你不想从事编写对象解析器的工作,你只是重新发明 JSON 或 YAML 等。所以你最好的选择是让你的格式符合已知的配置或对象表示法语言,然后用你正在使用的语言的一些库来解析它。通过非常小的修改,您上面描述的格式可以变成 HOCON,它是 JSON 的一个非常好的超集,并且语法更接近您正在使用的格式:
https://github.com/typesafehub/config/blob/master/HOCON.md
然后您可以使用 HOCON 解析库对其进行解析,瞧,您将拥有内存中的对象,您可以按照您喜欢的任何方式对其进行建模或存储。我相信 Qt 是基于 C++ 的?有一个用于 C 的 hocon 库,我不知道 C++,我猜你需要编写一个 Qt 插件来包装来自其他语言的 HOCON 解析。
另一种选择是使用 CSS->object 解析器,如下所示:
https://github.com/reworkcss/css
您可能需要根据您的需要进行分叉和修改。无论哪种方式,我猜测要集成到 Qt 应用程序中,您将需要一个插件来处理对命令行进程或其他代码模块的调用。
我知道两种可能:
- boost::spirit and here 你可以找到 boost::spirit 解析器框架的很好的介绍
- 我建议您自己编写 recursive descent parser
由于您的个性化 *.pss 不像 CSS 那样复杂(简单的括号等),我推荐 2.
我有一个类似 *.css(级联样式表)格式的文档,但它有自己的关键字。实际上它是一个个性化的 css(我称之为 *.pss),具有自己的标签和属性。在这里我有一段摘录:
/* CSS like style sheet file *.pss */
@include "otherStyleSheet.pss";
/* comment */
[propertyID="1230000"] {
fillColor : #f3f1ed;
minSize : 5;
lineWidth : 3;
}
/* sphere */
[propertyID="124???|123000"] {
lineType : dotted;
}
/* square */
[propertyID="125???"] {
lineType : thinline;
}
/* ring */
[propertyID="133???"] {
lineType : thickline;
[hasInnerRing=true] {
innerLineType : thinline;
}
}
我想很容易地解析它,是否已经有 Qt 的现成可用的东西?什么是最简单的方法?
由于 *.css 有自己的关键字,我对 CSS 解析器不感兴趣。
解析 *.pss 后我的进一步意图是将其属性存储在 Model 结构中。
Qt 中没有任何东西 public。您当然可以自由使用 Qt 的私有 CSS 解析器 - 您可以复制它并进行修改以满足您的需要。
参见 qtbase/src/gui/text/qcssparser_p.h
, in qtbase/src/gui/text
。
好消息是,对于上面显示的示例,修改非常小。 Qt 的 CSS 解析器已经支持 @import
,所以我们只有额外的一点语法,就是 nested selector syntax。如果没有该语法,您可以按原样使用 QCss::Parser
。解析器以灵活的方式编写,您无需担心正式的 CSS 关键字:它仍将允许您访问所有声明,无论它们从正式的 CSS 点是否有意义查看与否。
迭代解析树非常简单:
int main() {
QCss::Parser parser(pss);
QCss::StyleSheet styleSheet;
if (!parser.parse(&styleSheet))
return 1;
for (auto rule : styleSheet.styleRules) {
qDebug() << "** Rule **";
for (auto sel : rule.selectors) {
for (auto bSel : sel.basicSelectors)
qDebug() << bSel;
}
for (auto decl : rule.declarations)
qDebug() << decl;
}
}
输出是我们所期望的:
** Rule **
BasicSelector "propertyID"="1230000"
Declaration "fillColor" = '#f3f1ed' % QColor(ARGB 1, 0.952941, 0.945098, 0.929412)
Declaration "minSize" = '5' % 5
Declaration "lineWidth" = '3'
** Rule **
BasicSelector "propertyID"="124???|123000"
Declaration "lineType" = 'dotted'
** Rule **
BasicSelector "propertyID"="125???"
Declaration "lineType" = 'thinline'
** Rule **
BasicSelector "propertyID"="133???"
Declaration "lineType" = 'thickline'
我们必须自己为 QCss
类 实现调试流运算符:
QDebug operator<<(QDebug dbg, const QCss::AttributeSelector & sel) {
QDebugStateSaver saver(dbg);
dbg.noquote().nospace() << "\"" << sel.name << "\"";
switch (sel.valueMatchCriterium) {
case QCss::AttributeSelector::MatchEqual:
dbg << "="; break;
case QCss::AttributeSelector::MatchContains:
dbg << "~="; break;
case QCss::AttributeSelector::MatchBeginsWith:
dbg << "^="; break;
case QCss::AttributeSelector::NoMatch:
break;
}
if (sel.valueMatchCriterium != QCss::AttributeSelector::NoMatch && !sel.value.isEmpty())
dbg << "\"" << sel.value << "\"";
return dbg;
}
QDebug operator<<(QDebug dbg, const QCss::BasicSelector & sel) {
QDebugStateSaver saver(dbg);
dbg.noquote().nospace() << "BasicSelector";
if (!sel.elementName.isEmpty())
dbg << " #" << sel.elementName;
for (auto & id : sel.ids)
dbg << " id:" << id;
for (auto & aSel : sel.attributeSelectors)
dbg << " " << aSel;
return dbg;
}
在遍历声明时,QCss::parser
已经为我们解释了一些标准值,例如颜色、整数等
QDebug operator<<(QDebug dbg, const QCss::Declaration & decl) {
QDebugStateSaver saver(dbg);
dbg.noquote().nospace() << "Declaration";
dbg << " \"" << decl.d->property << "\" = ";
bool first = true;
for (auto value : decl.d->values) {
if (!first) dbg << ", ";
dbg << "\'" << value.toString() << "\'";
first = false;
}
if (decl.d->property == "fillColor")
dbg << " % " << decl.colorValue();
else if (decl.d->property == "minSize") {
int i;
if (decl.intValue(&i)) dbg << " % " << i;
}
return dbg;
}
最后,要解析的样板文件和样式表:
// https://github.com/KubaO/Whosebugn/tree/master/questions/css-like-parser-31583622
#include <QtGui>
#include <private/qcssparser_p.h>
const char pss[] =
"/* @include \"otherStyleSheet.pss\"; */ \
[propertyID=\"1230000\"] { \
fillColor : #f3f1ed; \
minSize : 5; \
lineWidth : 3; \
} \
\
/* sphere */ \
[propertyID=\"124???|123000\"] { \
lineType : dotted; \
} \
\
/* square */ \
[propertyID=\"125???\"] { \
lineType : thinline; \
} \
\
/* ring */ \
[propertyID=\"133???\"] { \
lineType : thickline; \
/*[hasInnerRing=true] { \
innerLineType : thinline; \
}*/ \
}";
对嵌套的支持selectors/rules可以通过修改解析器源来实现。使 Parser::parseRuleset
递归所需的更改非常小。我将把它留作 reader :)
总而言之,我认为重用现有的解析器比推出自己的解析器要容易得多,尤其是当您的用户不可避免地希望您支持越来越多的 CSS 规范时。
好吧,我猜你不想从事编写对象解析器的工作,你只是重新发明 JSON 或 YAML 等。所以你最好的选择是让你的格式符合已知的配置或对象表示法语言,然后用你正在使用的语言的一些库来解析它。通过非常小的修改,您上面描述的格式可以变成 HOCON,它是 JSON 的一个非常好的超集,并且语法更接近您正在使用的格式:
https://github.com/typesafehub/config/blob/master/HOCON.md
然后您可以使用 HOCON 解析库对其进行解析,瞧,您将拥有内存中的对象,您可以按照您喜欢的任何方式对其进行建模或存储。我相信 Qt 是基于 C++ 的?有一个用于 C 的 hocon 库,我不知道 C++,我猜你需要编写一个 Qt 插件来包装来自其他语言的 HOCON 解析。
另一种选择是使用 CSS->object 解析器,如下所示: https://github.com/reworkcss/css
您可能需要根据您的需要进行分叉和修改。无论哪种方式,我猜测要集成到 Qt 应用程序中,您将需要一个插件来处理对命令行进程或其他代码模块的调用。
我知道两种可能:
- boost::spirit and here 你可以找到 boost::spirit 解析器框架的很好的介绍
- 我建议您自己编写 recursive descent parser
由于您的个性化 *.pss 不像 CSS 那样复杂(简单的括号等),我推荐 2.