mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
[Refactoring] Move MemberwiseInitLocalRefactoring to its own file
This commit is contained in:
159
lib/Refactoring/MemberwiseInitLocalRefactoring.cpp
Normal file
159
lib/Refactoring/MemberwiseInitLocalRefactoring.cpp
Normal file
@@ -0,0 +1,159 @@
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This source file is part of the Swift.org open source project
|
||||
//
|
||||
// Copyright (c) 2014 - 2023 Apple Inc. and the Swift project authors
|
||||
// Licensed under Apache License v2.0 with Runtime Library Exception
|
||||
//
|
||||
// See https://swift.org/LICENSE.txt for license information
|
||||
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "RefactoringActions.h"
|
||||
|
||||
using namespace swift::refactoring;
|
||||
|
||||
namespace {
|
||||
struct MemberwiseParameter {
|
||||
CharSourceRange NameRange;
|
||||
Type MemberType;
|
||||
Expr *DefaultExpr;
|
||||
|
||||
MemberwiseParameter(CharSourceRange nameRange, Type type, Expr *initialExpr)
|
||||
: NameRange(nameRange), MemberType(type), DefaultExpr(initialExpr) {}
|
||||
};
|
||||
} // namespace
|
||||
|
||||
static void generateMemberwiseInit(SourceEditConsumer &EditConsumer,
|
||||
SourceManager &SM,
|
||||
ArrayRef<MemberwiseParameter> memberVector,
|
||||
SourceLoc targetLocation) {
|
||||
|
||||
EditConsumer.accept(SM, targetLocation, "\ninternal init(");
|
||||
auto insertMember = [&SM](const MemberwiseParameter &memberData,
|
||||
raw_ostream &OS, bool wantsSeparator) {
|
||||
{
|
||||
OS << SM.extractText(memberData.NameRange) << ": ";
|
||||
// Unconditionally print '@escaping' if we print out a function type -
|
||||
// the assignments we generate below will escape this parameter.
|
||||
if (isa<AnyFunctionType>(memberData.MemberType->getCanonicalType())) {
|
||||
OS << "@" << TypeAttributes::getAttrName(TAK_escaping) << " ";
|
||||
}
|
||||
OS << memberData.MemberType.getString();
|
||||
}
|
||||
|
||||
bool HasAddedDefault = false;
|
||||
if (auto *expr = memberData.DefaultExpr) {
|
||||
if (expr->getSourceRange().isValid()) {
|
||||
auto range = Lexer::getCharSourceRangeFromSourceRange(
|
||||
SM, expr->getSourceRange());
|
||||
OS << " = " << SM.extractText(range);
|
||||
HasAddedDefault = true;
|
||||
}
|
||||
}
|
||||
if (!HasAddedDefault && memberData.MemberType->isOptional()) {
|
||||
OS << " = nil";
|
||||
}
|
||||
|
||||
if (wantsSeparator) {
|
||||
OS << ", ";
|
||||
}
|
||||
};
|
||||
|
||||
// Process the initial list of members, inserting commas as appropriate.
|
||||
std::string Buffer;
|
||||
llvm::raw_string_ostream OS(Buffer);
|
||||
for (const auto &memberData : llvm::enumerate(memberVector)) {
|
||||
bool wantsSeparator = (memberData.index() != memberVector.size() - 1);
|
||||
insertMember(memberData.value(), OS, wantsSeparator);
|
||||
}
|
||||
|
||||
// Synthesize the body.
|
||||
OS << ") {\n";
|
||||
for (auto &member : memberVector) {
|
||||
// self.<property> = <property>
|
||||
auto name = SM.extractText(member.NameRange);
|
||||
OS << "self." << name << " = " << name << "\n";
|
||||
}
|
||||
OS << "}\n";
|
||||
|
||||
// Accept the entire edit.
|
||||
EditConsumer.accept(SM, targetLocation, OS.str());
|
||||
}
|
||||
|
||||
static SourceLoc
|
||||
collectMembersForInit(ResolvedCursorInfoPtr CursorInfo,
|
||||
SmallVectorImpl<MemberwiseParameter> &memberVector) {
|
||||
auto ValueRefInfo = dyn_cast<ResolvedValueRefCursorInfo>(CursorInfo);
|
||||
if (!ValueRefInfo || !ValueRefInfo->getValueD())
|
||||
return SourceLoc();
|
||||
|
||||
NominalTypeDecl *nominalDecl =
|
||||
dyn_cast<NominalTypeDecl>(ValueRefInfo->getValueD());
|
||||
if (!nominalDecl || nominalDecl->getStoredProperties().empty() ||
|
||||
ValueRefInfo->isRef()) {
|
||||
return SourceLoc();
|
||||
}
|
||||
|
||||
SourceLoc bracesStart = nominalDecl->getBraces().Start;
|
||||
if (!bracesStart.isValid())
|
||||
return SourceLoc();
|
||||
|
||||
SourceLoc targetLocation = bracesStart.getAdvancedLoc(1);
|
||||
if (!targetLocation.isValid())
|
||||
return SourceLoc();
|
||||
|
||||
SourceManager &SM = nominalDecl->getASTContext().SourceMgr;
|
||||
|
||||
for (auto member : nominalDecl->getMemberwiseInitProperties()) {
|
||||
auto varDecl = dyn_cast<VarDecl>(member);
|
||||
if (!varDecl) {
|
||||
continue;
|
||||
}
|
||||
if (varDecl->getAttrs().hasAttribute<LazyAttr>()) {
|
||||
// Exclude lazy members from the memberwise initializer. This is
|
||||
// inconsistent with the implicitly synthesized memberwise initializer but
|
||||
// we think it makes more sense because otherwise the lazy variable's
|
||||
// initializer gets evaluated eagerly.
|
||||
continue;
|
||||
}
|
||||
|
||||
auto patternBinding = varDecl->getParentPatternBinding();
|
||||
if (!patternBinding)
|
||||
continue;
|
||||
|
||||
const auto i = patternBinding->getPatternEntryIndexForVarDecl(varDecl);
|
||||
Expr *defaultInit = nullptr;
|
||||
if (patternBinding->isExplicitlyInitialized(i) ||
|
||||
patternBinding->isDefaultInitializable()) {
|
||||
defaultInit = patternBinding->getOriginalInit(i);
|
||||
}
|
||||
|
||||
auto NameRange =
|
||||
Lexer::getCharSourceRangeFromSourceRange(SM, varDecl->getNameLoc());
|
||||
memberVector.emplace_back(NameRange, varDecl->getTypeInContext(),
|
||||
defaultInit);
|
||||
}
|
||||
|
||||
return targetLocation;
|
||||
}
|
||||
|
||||
bool RefactoringActionMemberwiseInitLocalRefactoring::isApplicable(
|
||||
ResolvedCursorInfoPtr Tok, DiagnosticEngine &Diag) {
|
||||
|
||||
SmallVector<MemberwiseParameter, 8> memberVector;
|
||||
return collectMembersForInit(Tok, memberVector).isValid();
|
||||
}
|
||||
|
||||
bool RefactoringActionMemberwiseInitLocalRefactoring::performChange() {
|
||||
|
||||
SmallVector<MemberwiseParameter, 8> memberVector;
|
||||
SourceLoc targetLocation = collectMembersForInit(CursorInfo, memberVector);
|
||||
if (targetLocation.isInvalid())
|
||||
return true;
|
||||
|
||||
generateMemberwiseInit(EditConsumer, SM, memberVector, targetLocation);
|
||||
|
||||
return false;
|
||||
}
|
||||
Reference in New Issue
Block a user