7#include <customwidgetsinfo.h>
14#include <QtCore/qdir.h>
15#include <QtCore/qfileinfo.h>
16#include <QtCore/qtextstream.h>
22using namespace Qt::StringLiterals;
61static void insertClass(
const QString &module,
const QString &className,
62 WriteImports::ClassesPerModule *c)
64 auto usedIt = c->find(module);
65 if (usedIt == c->end())
66 c->insert(module, {className});
67 else if (!usedIt.value().contains(className))
68 usedIt.value().append(className);
74 std::sort(classList.begin(), classList.end());
76 const qsizetype size = classList.size();
79 for (qsizetype i = 0; i < size; ++i) {
81 str << (i % 4 == 0 ?
",\n " :
", ");
82 QString name = language::fixClassName(classList.at(i));
83 if (
const auto dotPos = name.indexOf(u'.'); dotPos != -1)
84 name.truncate(dotPos);
91static void formatClasses(QTextStream &str,
const WriteImports::ClassesPerModule &c,
92 bool useStarImports =
false,
93 const QByteArray &modulePrefix = {})
95 for (
auto it = c.cbegin(), end = c.cend(); it != end; ++it) {
96 str <<
"from " << modulePrefix << it.key() <<
" import ";
98 str <<
"* # type: ignore";
100 formatImportClasses(str, it.value());
108 for (
const auto &e : classInfoEntries())
109 m_classToModule.insert(QLatin1StringView(e.klass), QLatin1StringView(e.module));
116 auto &output = uic()->output();
117 const bool useStarImports = uic()->driver()->option().useStarImports;
119 const QByteArray qtPrefix = QByteArrayLiteral(
"PySide")
120 + QByteArray::number(QT_VERSION_MAJOR) +
'.';
122 formatClasses(output, m_qtClasses, useStarImports, qtPrefix);
124 if (!m_customWidgets.isEmpty() || !m_plainCustomWidgets.isEmpty()) {
126 formatClasses(output, m_customWidgets, useStarImports);
127 for (
const auto &w : m_plainCustomWidgets)
128 output <<
"import " << w <<
'\n';
131 if (
auto *resources = node->elementResources()) {
132 const auto &includes = resources->elementInclude();
133 for (
auto *include : includes) {
134 if (include->hasAttributeLocation())
135 writeResourceImport(include->attributeLocation());
141QString
WriteImports::resourceAbsolutePath(QString resource)
const
146 const auto &options = uic()->option();
147 if (!options.inputFile.isEmpty() && !options.pythonRoot.isEmpty()) {
148 resource = QDir::cleanPath(QFileInfo(options.inputFile).canonicalPath() + u'/' + resource);
149 if (resource.size() > options.pythonRoot.size())
150 resource.remove(0, options.pythonRoot.size() + 1);
153 while (resource.startsWith(u"../"))
154 resource.remove(0, 3);
155 resource.replace(u'/', u'.');
159void WriteImports::writeResourceImport(
const QString &module)
161 const auto &options = uic()->option();
162 auto &str = uic()->output();
164 QString resource = QDir::cleanPath(module);
165 if (resource.endsWith(u".qrc"))
167 const qsizetype basePos = resource.lastIndexOf(u'/') + 1;
170 if (options.rcPrefix)
171 resource.insert(basePos, u"rc_");
173 resource.append(u"_rc");
175 switch (options.pythonResourceImport) {
176 case Option::PythonResourceImport::Default:
177 str <<
"import " << QStringView{resource}.sliced(basePos) <<
'\n';
179 case Option::PythonResourceImport::FromDot:
180 str <<
"from . import " << QStringView{resource}.sliced(basePos) <<
'\n';
182 case Option::PythonResourceImport::Absolute:
183 str <<
"import " << resourceAbsolutePath(resource) <<
'\n';
188void WriteImports::doAdd(
const QString &className,
const DomCustomWidget *dcw)
191 if (cwi->extends(className,
"QListWidget"))
193 else if (cwi->extends(className,
"QTreeWidget"))
195 else if (cwi->extends(className,
"QTableWidget"))
198 if (dcw !=
nullptr) {
199 addPythonCustomWidget(className, dcw);
203 if (!addQtClass(className))
204 qWarning(
"WriteImports::add(): Unknown Qt class %s",
qPrintable(className));
210 if (className == u"QVariant" || className == u"Qt")
213 const auto moduleIt = m_classToModule.constFind(className);
214 const bool result = moduleIt != m_classToModule.cend();
216 insertClass(moduleIt.value(), className, &m_qtClasses);
220void WriteImports::addPythonCustomWidget(
const QString &className,
const DomCustomWidget *node)
222 if (addQtClass(className))
227 if (!node->elementHeader() || node->elementHeader()->text().isEmpty()) {
228 m_plainCustomWidgets.append(className);
230 QString modulePath = node->elementHeader()->text();
232 modulePath.replace(u'/', u'.');
234 if (modulePath.endsWith(
".h"_L1, Qt::CaseInsensitive))
236 else if (modulePath.endsWith(
".hh"_L1))
238 else if (modulePath.endsWith(
".hpp"_L1))
240 insertClass(modulePath, className, &m_customWidgets);
246 switch (node->kind()) {
247 case DomProperty::Enum:
248 addEnumBaseClass(node->elementEnum());
250 case DomProperty::Set:
251 addEnumBaseClass(node->elementSet());
263 const auto colonPos = v.indexOf(u"::");
265 const QString base = v.left(colonPos);
266 if (base.startsWith(u'Q') && base != u"Qt")
void acceptUI(DomUI *node) override
void acceptProperty(DomProperty *node) override
WriteIncludesBase(Uic *uic)
void acceptUI(DomUI *node) override
void acceptProperty(DomProperty *node) override
static void formatImportClasses(QTextStream &str, QStringList classList)
static void insertClass(const QString &module, const QString &className, WriteImports::ClassesPerModule *c)
static void formatClasses(QTextStream &str, const WriteImports::ClassesPerModule &c, bool useStarImports=false, const QByteArray &modulePrefix={})
static WriteImports::ClassesPerModule defaultClasses()
Combined button and popup list for selecting options.
const QString & asString(const QString &s)
#define qPrintable(string)
#define QStringLiteral(str)