1 | /**************************************************************************** |
---|---|
2 | ** |
3 | ** Copyright (C) 2019 The Qt Company Ltd. |
4 | ** Contact: https://www.qt.io/licensing/ |
5 | ** |
6 | ** This file is part of the tools applications of the Qt Toolkit. |
7 | ** |
8 | ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ |
9 | ** Commercial License Usage |
10 | ** Licensees holding valid commercial Qt licenses may use this file in |
11 | ** accordance with the commercial license agreement provided with the |
12 | ** Software or, alternatively, in accordance with the terms contained in |
13 | ** a written agreement between you and The Qt Company. For licensing terms |
14 | ** and conditions see https://www.qt.io/terms-conditions. For further |
15 | ** information use the contact form at https://www.qt.io/contact-us. |
16 | ** |
17 | ** GNU General Public License Usage |
18 | ** Alternatively, this file may be used under the terms of the GNU |
19 | ** General Public License version 3 as published by the Free Software |
20 | ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT |
21 | ** included in the packaging of this file. Please review the following |
22 | ** information to ensure the GNU General Public License requirements will |
23 | ** be met: https://www.gnu.org/licenses/gpl-3.0.html. |
24 | ** |
25 | ** $QT_END_LICENSE$ |
26 | ** |
27 | ****************************************************************************/ |
28 | |
29 | #include "pythonwriteimports.h" |
30 | |
31 | #include <customwidgetsinfo.h> |
32 | #include <option.h> |
33 | #include <uic.h> |
34 | |
35 | #include <ui4.h> |
36 | |
37 | #include <QtCore/qtextstream.h> |
38 | |
39 | QT_BEGIN_NAMESPACE |
40 | |
41 | static const char *standardImports = |
42 | R"I(from PySide2.QtCore import * |
43 | from PySide2.QtGui import * |
44 | from PySide2.QtWidgets import * |
45 | )I"; |
46 | |
47 | // Change the name of a qrc file "dir/foo.qrc" file to the Python |
48 | // module name "foo_rc" according to project conventions. |
49 | static QString pythonResource(QString resource) |
50 | { |
51 | const int lastSlash = resource.lastIndexOf(QLatin1Char('/')); |
52 | if (lastSlash != -1) |
53 | resource.remove(0, lastSlash + 1); |
54 | if (resource.endsWith(QLatin1String(".qrc"))) { |
55 | resource.chop(4); |
56 | resource.append(QLatin1String("_rc")); |
57 | } |
58 | return resource; |
59 | } |
60 | |
61 | namespace Python { |
62 | |
63 | WriteImports::WriteImports(Uic *uic) : m_uic(uic) |
64 | { |
65 | } |
66 | |
67 | void WriteImports::acceptUI(DomUI *node) |
68 | { |
69 | auto &output = m_uic->output(); |
70 | output << standardImports << '\n'; |
71 | if (auto customWidgets = node->elementCustomWidgets()) { |
72 | TreeWalker::acceptCustomWidgets(customWidgets); |
73 | output << '\n'; |
74 | } |
75 | |
76 | if (auto resources = node->elementResources()) { |
77 | const auto includes = resources->elementInclude(); |
78 | for (auto include : includes) { |
79 | if (include->hasAttributeLocation()) |
80 | writeImport(pythonResource(include->attributeLocation())); |
81 | } |
82 | output << '\n'; |
83 | } |
84 | } |
85 | |
86 | void WriteImports::writeImport(const QString &module) |
87 | { |
88 | |
89 | if (m_uic->option().fromImports) |
90 | m_uic->output() << "from . "; |
91 | m_uic->output() << "import "<< module << '\n'; |
92 | } |
93 | |
94 | QString WriteImports::qtModuleOf(const DomCustomWidget *node) const |
95 | { |
96 | if (m_uic->customWidgetsInfo()->extends(node->elementClass(), QLatin1String("QAxWidget"))) |
97 | return QStringLiteral("QtAxContainer"); |
98 | if (const auto headerElement = node->elementHeader()) { |
99 | const auto &header = headerElement->text(); |
100 | if (header.startsWith(QLatin1String("Qt"))) { |
101 | const int slash = header.indexOf(QLatin1Char('/')); |
102 | if (slash != -1) |
103 | return header.left(slash); |
104 | } |
105 | } |
106 | return QString(); |
107 | } |
108 | |
109 | void WriteImports::acceptCustomWidget(DomCustomWidget *node) |
110 | { |
111 | const auto &className = node->elementClass(); |
112 | if (className.contains(QLatin1String("::"))) |
113 | return; // Exclude namespaced names (just to make tests pass). |
114 | const QString &importModule = qtModuleOf(node); |
115 | auto &output = m_uic->output(); |
116 | // For starting importing PySide2 modules |
117 | if (!importModule.isEmpty()) { |
118 | output << "from "; |
119 | if (importModule.startsWith(QLatin1String("Qt"))) |
120 | output << "PySide2."; |
121 | output << importModule; |
122 | if (!className.isEmpty()) |
123 | output << " import "<< className << "\n\n"; |
124 | } else { |
125 | // When the elementHeader is not set, we know it's the continuation |
126 | // of a PySide2 import or a normal import of another module. |
127 | if (!node->elementHeader() || node->elementHeader()->text().isEmpty()) { |
128 | output << "import "<< className << '\n'; |
129 | } else { // When we do have elementHeader, we know it's a relative import. |
130 | QString modulePath = node->elementHeader()->text(); |
131 | // Replace the '/' by '.' |
132 | modulePath.replace(QLatin1Char('/'), QLatin1Char('.')); |
133 | // '.h' is added by default on headers for <customwidget> |
134 | if (modulePath.endsWith(QLatin1String(".h"))) |
135 | modulePath.chop(2); |
136 | output << "from "<< modulePath << " import "<< className << '\n'; |
137 | } |
138 | } |
139 | } |
140 | |
141 | } // namespace Python |
142 | |
143 | QT_END_NAMESPACE |
144 |