Non c'è nulla di pubblico in Qt. Ovviamente sei libero di utilizzare il parser CSS privato di Qt: puoi copiarlo e modificarlo in base alle tue esigenze.
Vedere qtbase/src/gui/text/qcssparser_p.h
, in qtbase/src/gui/text
.
La buona notizia è che per l'esempio che hai mostrato sopra, le modifiche sarebbero molto minori. Il parser CSS di Qt supporta già @import
, quindi abbiamo solo un ulteriore bit di sintassi che possiedi è il nested selector syntax. Senza questa sintassi, è possibile utilizzare QCss::Parser
così com'è. Il parser è stato scritto in modo flessibile, in cui non è necessario preoccuparsi delle parole chiave formali CSS: consente comunque di accedere a tutte le dichiarazioni, indipendentemente dal loro punto di vista formale o meno.
Iterando l'albero sintattico è così semplice come si arriva:
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;
}
}
L'uscita è quello che ci si aspetterebbe:
** 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'
Dobbiamo implementare gli operatori del flusso di debug per QCss
classi noi stessi :
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;
}
Quando si attraversa la dichiarazione, il QCss::parser
interpreta già per noi alcuni valori standard, ad es. i colori, numeri interi, ecc
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;
}
Infine, il boilerplate e il foglio di stile da analizzare:
// https://github.com/KubaO/stackoverflown/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; \
}*/ \
}";
supporto per selettori nidificate/regole può essere implementata modificando la fonte parser. La modifica necessaria per rendere ricorsiva Parser::parseRuleset
è molto minore. Lascerò questo come esercizio per il lettore :)
Tutto sommato, penserei che riutilizzare il parser esistente sia molto più facile che farlo da solo, soprattutto perché gli utenti vorranno inevitabilmente supportare di più e più delle specifiche CSS.
Non sono a conoscenza di nulla disponibile. Sarebbe un vero e proprio CSS o CSS di cui hai il controllo? –
@FrankOsterfeld Mi spiace, non conosco la differenza fuori dagli schemi e il controllo? Ho modificato la mia domanda .. Thnx –
Ciò che Frank intende è: il CSS è una parte della tua applicazione, in modo che tu possa modificarla e gestirla, o provenga da fonti potenzialmente dannose su Internet? Ricorda che se il tuo parser CSS ha dei bug, i CSS dannosi possono sfruttarli e prendere in carico la tua applicazione. In un codice C++ (o C) meno che moderno, tali bug sono più la norma che l'eccezione. Il parser di Qt non è progettato per essere resistente ai CSS dannosi, non è mai stato progettato per accettare input casuali da Internet. Ha un sacco di buchi di sicurezza, ne sono sicuro. –