24collectNestedIds(
const QQmlLSUtils::ItemLocation &item)
27 item.domItem.goToFile(item.domItem.canonicalFilePath()).ownerAs<QQmlJS::Dom::QmlFile>();
28 const QString code = filePtr ? filePtr->code() : QString();
30 QList<std::pair<QString, QQmlJS::SourceLocation>> innerIds;
31 FileLocations::visitTree(
33 [&code, &innerIds](
const auto &,
const FileLocations::Tree &t) ->
bool {
34 const auto idNameLoc = t->info().regions[FileLocationRegion::IdNameRegion];
35 if (idNameLoc.isValid()) {
36 innerIds.append({ code.mid(idNameLoc.begin(), idNameLoc.length), idNameLoc });
45 CodeActions codeActions;
47 for (
const Diagnostic &diagnostic : diagnostics) {
48 if (!diagnostic.data.has_value())
51 const auto &data = diagnostic.data.value();
53 int version = data[u"version"].toInt();
54 QJsonArray suggestions = data[u"suggestions"].toArray();
56 QList<WorkspaceEdit::DocumentChange> edits;
58 for (
const QJsonValue &suggestion : std::as_const(suggestions)) {
59 message += suggestion[u"message"_s].toString() + u'\n';
60 const auto &documentEdits = suggestion[u"documentEdits"_s].toArray();
61 TextDocumentEdit textDocEdit;
62 for (
const auto &documentEdit : documentEdits) {
66 static_cast<
unsigned>(documentEdit[u"lspBeginLine"].toDouble()),
67 static_cast<
unsigned>(documentEdit[u"lspBeginCharacter"].toDouble()) },
68 Position{
static_cast<
unsigned>(documentEdit[u"lspEndLine"].toDouble()),
69 static_cast<
unsigned>(documentEdit[u"lspEndCharacter"].toDouble()) }
72 textEdit.newText = documentEdit[u"replacement"_s].toString().toUtf8();
73 QString filename = documentEdit[u"filename"_s].toString();
74 textDocEdit.textDocument = { { filename.toUtf8() }, version };
75 textDocEdit.edits.append(textEdit);
77 edits.append(textDocEdit);
81 edit.documentChanges = edits;
85 action.kind = CodeActionKind::QuickFix;
87 action.title = message.toUtf8();
89 codeActions.append(action);
97 const QString todo = Tr::tr(
"TODO:");
98 const QString prefix =
"//"_L1 + QString(2 + todo.size(), u' ');
100 QString comment =
"// "_L1 + todo + u' '
101 + Tr::tr(
"Move position bindings from the component to the Loader.") + u'\n'
103 + Tr::tr(
"Check all uses of 'parent' inside the root element of the component.") + u'\n';
105 if (!maybeId.isEmpty()) {
107 + Tr::tr(
"Rename all outer uses of the id \"%1\" to \"%2.item\".").arg(maybeId, loaderId) + u'\n';
109 for (
const auto &id : nestedIds) {
111 + Tr::tr(
"Rename all outer uses of the id \"%1\" to \"%2.item.%1\".").arg(id.first, loaderId) + u'\n';
114 return { { pos, pos }, comment.toUtf8() };
119 const QString componentOpen = QString::fromLatin1(
"Component {\n"
122 const QString componentClose = QString::fromLatin1(
"\n}\n");
123 return { TextEdit{ { itemRange.start, itemRange.start }, componentOpen.toUtf8() },
124 TextEdit{ { itemRange.end, itemRange.end }, componentClose.toUtf8() } };
141 const QLatin1StringView nestedIdPrefix(
"inner_");
142 QString idAliases = QString::fromLatin1(
"\n");
143 for (
auto &id : nestedIds) {
144 idAliases += QString::fromLatin1(
"property alias %1: %2%1\n").arg(id.first, nestedIdPrefix);
149 const auto posAfterLBrace =
150 Position{
static_cast<
unsigned int>(
static_cast<
int>(openingBrace.startLine - 1)),
151 static_cast<
unsigned int>(
152 static_cast<
int>(openingBrace.startColumn)) };
154 edits.append(TextEdit{ { posAfterLBrace, posAfterLBrace }, idAliases.toUtf8() });
157 for (
auto &id : nestedIds) {
159 Position{
static_cast<
unsigned int>(
static_cast<
int>(id.second.startLine - 1)),
160 static_cast<
unsigned int>(
static_cast<
int>(id.second.startColumn - 1)) };
161 edits.append(TextEdit{ { idPos, idPos }, nestedIdPrefix.toUtf8() });
168 const auto generateId = [&item](
const QString &base) -> QString {
169 const auto ids = item.domItem.component().field(Fields::ids).keys();
170 if (!ids.contains(base)) {
174 for (; extraNumber < ids.size(); ++extraNumber) {
175 QString id = base + QString::number(extraNumber);
176 if (!ids.contains(id)) {
180 return base + QString::number(extraNumber);
183 const auto objId = item.domItem.idStr();
184 const auto objName = objId.isEmpty() ? item.domItem.name() : objId;
185 const QString componentId = generateId(QLatin1StringView(
"component_") + objName);
186 const QString loaderId = generateId(QLatin1StringView(
"loader_") + objName);
188 const auto itemRange = QQmlLSUtils::qmlLocationToLspLocation(
189 QQmlLSUtils::Location::tryFrom(item.domItem.canonicalFilePath(),
190 item.fileLocation->info().fullRegion, item.domItem)
191 .value_or(QQmlLSUtils::Location{}));
193 QList<std::pair<QString, QQmlJS::SourceLocation>> nestedIds = collectNestedIds(item);
194 if (!objId.isEmpty()) {
197 Q_ASSERT(nestedIds.front().first == objId);
198 nestedIds.removeFirst();
202 edits << todoComment(itemRange.start, loaderId, objId, nestedIds)
203 << exposeNestedIds(item.fileLocation->info().regions[FileLocationRegion::LeftBraceRegion],
205 << wrapIntoComponent(itemRange, componentId)
206 << addLoader(itemRange.end, loaderId, componentId);
213 if (item.domItem.internalKind() != DomType::ScriptIdentifierExpression) {
216 auto parentObject = item.domItem.qmlObject();
217 const auto objectLoc = FileLocations::treeOf(parentObject);
219 if (item.fileLocation->info().fullRegion.begin() != objectLoc->info().fullRegion.begin()) {
223 return QQmlLSUtils::ItemLocation{ parentObject, objectLoc };
228 const QQmlLSUtils::ItemLocation &item)
230 if (item.domItem.internalKind() != DomType::QmlObject) {
233 if (
auto qmlObject = qmlObjectDefinedAt(item)) {
234 return wrapComponentInLoader(textDocument, *qmlObject);
239 if (item.domItem == item.domItem.component().field(Fields::objects).index(0)) {
243 if (item.domItem.canonicalPath().last() == Path::fromField(Fields::value)) {
248 TextDocumentEdit textDocEdit{ textDocument, wrapInLoaderTextEdits(item) };
251 edit.documentChanges = { textDocEdit };
254 action.kind = CodeActionKind::RefactorRewrite;
255 action.title = Tr::tr(
"Wrap Component in Loader").toUtf8();
282 CodeActions codeActions;
283 codeActions.append(quickfixes(request->m_parameters.context.diagnostics));
285 const auto &maybeDoc = tryOpenDocument(request->m_parameters.textDocument.uri);
286 if (!maybeDoc.has_value()) {
287 qCWarning(lsCodeActionSupport) << maybeDoc.error().message;
288 return request->m_response.sendResponse(codeActions);
291 const auto &itemsFound = tryLocateItems(maybeDoc.value(), request->m_parameters.range.start);
292 if (!itemsFound.has_value()) {
293 qCWarning(lsCodeActionSupport) << itemsFound.error().message;
294 return request->m_response.sendResponse(codeActions);
297 codeActions.append(refactorings(
298 { request->m_parameters.textDocument, maybeDoc.value().snapshot.validDocVersion },
299 itemsFound.value().front()));
301 request->m_response.sendResponse(codeActions);
static TextEdit todoComment(const Position &pos, const QString &loaderId, const QString &maybeId, const QList< std::pair< QString, QQmlJS::SourceLocation > > &nestedIds)