13af6ab5fSopenharmony_ci/**
23af6ab5fSopenharmony_ci * Copyright (c) 2021-2024 Huawei Device Co., Ltd.
33af6ab5fSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
43af6ab5fSopenharmony_ci * you may not use this file except in compliance with the License.
53af6ab5fSopenharmony_ci * You may obtain a copy of the License at
63af6ab5fSopenharmony_ci *
73af6ab5fSopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0
83af6ab5fSopenharmony_ci *
93af6ab5fSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
103af6ab5fSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
113af6ab5fSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
123af6ab5fSopenharmony_ci * See the License for the specific language governing permissions and
133af6ab5fSopenharmony_ci * limitations under the License.
143af6ab5fSopenharmony_ci */
153af6ab5fSopenharmony_ci
163af6ab5fSopenharmony_ci#include "classDefinition.h"
173af6ab5fSopenharmony_ci
183af6ab5fSopenharmony_ci#include "checker/TSchecker.h"
193af6ab5fSopenharmony_ci#include "checker/ETSchecker.h"
203af6ab5fSopenharmony_ci#include "compiler/core/ETSGen.h"
213af6ab5fSopenharmony_ci#include "compiler/core/pandagen.h"
223af6ab5fSopenharmony_ci#include "ir/astDump.h"
233af6ab5fSopenharmony_ci#include "ir/srcDump.h"
243af6ab5fSopenharmony_ci#include "ir/base/classStaticBlock.h"
253af6ab5fSopenharmony_ci#include "ir/base/methodDefinition.h"
263af6ab5fSopenharmony_ci#include "ir/base/scriptFunction.h"
273af6ab5fSopenharmony_ci#include "ir/expressions/identifier.h"
283af6ab5fSopenharmony_ci#include "ir/ts/tsClassImplements.h"
293af6ab5fSopenharmony_ci
303af6ab5fSopenharmony_cinamespace ark::es2panda::ir {
313af6ab5fSopenharmony_ciconst FunctionExpression *ClassDefinition::Ctor() const
323af6ab5fSopenharmony_ci{
333af6ab5fSopenharmony_ci    return ctor_ != nullptr ? ctor_->Value()->AsFunctionExpression() : nullptr;
343af6ab5fSopenharmony_ci}
353af6ab5fSopenharmony_ci
363af6ab5fSopenharmony_cibool ClassDefinition::HasPrivateMethod() const
373af6ab5fSopenharmony_ci{
383af6ab5fSopenharmony_ci    return std::any_of(body_.cbegin(), body_.cend(), [](auto *element) {
393af6ab5fSopenharmony_ci        return element->IsMethodDefinition() && element->AsClassElement()->IsPrivateElement();
403af6ab5fSopenharmony_ci    });
413af6ab5fSopenharmony_ci}
423af6ab5fSopenharmony_ci
433af6ab5fSopenharmony_cibool ClassDefinition::HasComputedInstanceField() const
443af6ab5fSopenharmony_ci{
453af6ab5fSopenharmony_ci    return std::any_of(body_.cbegin(), body_.cend(), [](auto *element) {
463af6ab5fSopenharmony_ci        return element->IsClassProperty() && element->AsClassElement()->IsComputed() &&
473af6ab5fSopenharmony_ci               !(element->AsClassElement()->Modifiers() & ir::ModifierFlags::STATIC);
483af6ab5fSopenharmony_ci    });
493af6ab5fSopenharmony_ci}
503af6ab5fSopenharmony_ci
513af6ab5fSopenharmony_cibool ClassDefinition::HasMatchingPrivateKey(const util::StringView &name) const
523af6ab5fSopenharmony_ci{
533af6ab5fSopenharmony_ci    return std::any_of(body_.cbegin(), body_.cend(), [&name](auto *element) {
543af6ab5fSopenharmony_ci        return element->AsClassElement()->IsPrivateElement() && element->AsClassElement()->Id()->Name() == name;
553af6ab5fSopenharmony_ci    });
563af6ab5fSopenharmony_ci}
573af6ab5fSopenharmony_ci
583af6ab5fSopenharmony_civoid ClassDefinition::TransformChildren(const NodeTransformer &cb, std::string_view transformationName)
593af6ab5fSopenharmony_ci{
603af6ab5fSopenharmony_ci    if (ident_ != nullptr) {
613af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(ident_); ident_ != transformedNode) {
623af6ab5fSopenharmony_ci            ident_->SetTransformedNode(transformationName, transformedNode);
633af6ab5fSopenharmony_ci            ident_ = transformedNode->AsIdentifier();
643af6ab5fSopenharmony_ci        }
653af6ab5fSopenharmony_ci    }
663af6ab5fSopenharmony_ci
673af6ab5fSopenharmony_ci    if (typeParams_ != nullptr) {
683af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(typeParams_); typeParams_ != transformedNode) {
693af6ab5fSopenharmony_ci            typeParams_->SetTransformedNode(transformationName, transformedNode);
703af6ab5fSopenharmony_ci            typeParams_ = transformedNode->AsTSTypeParameterDeclaration();
713af6ab5fSopenharmony_ci        }
723af6ab5fSopenharmony_ci    }
733af6ab5fSopenharmony_ci
743af6ab5fSopenharmony_ci    if (superClass_ != nullptr) {
753af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(superClass_); superClass_ != transformedNode) {
763af6ab5fSopenharmony_ci            superClass_->SetTransformedNode(transformationName, transformedNode);
773af6ab5fSopenharmony_ci            superClass_ = transformedNode->AsExpression();
783af6ab5fSopenharmony_ci        }
793af6ab5fSopenharmony_ci    }
803af6ab5fSopenharmony_ci
813af6ab5fSopenharmony_ci    if (superTypeParams_ != nullptr) {
823af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(superTypeParams_); superTypeParams_ != transformedNode) {
833af6ab5fSopenharmony_ci            superTypeParams_->SetTransformedNode(transformationName, transformedNode);
843af6ab5fSopenharmony_ci            superTypeParams_ = transformedNode->AsTSTypeParameterInstantiation();
853af6ab5fSopenharmony_ci        }
863af6ab5fSopenharmony_ci    }
873af6ab5fSopenharmony_ci
883af6ab5fSopenharmony_ci    for (auto *&it : implements_) {
893af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(it); it != transformedNode) {
903af6ab5fSopenharmony_ci            it->SetTransformedNode(transformationName, transformedNode);
913af6ab5fSopenharmony_ci            it = transformedNode->AsTSClassImplements();
923af6ab5fSopenharmony_ci        }
933af6ab5fSopenharmony_ci    }
943af6ab5fSopenharmony_ci
953af6ab5fSopenharmony_ci    if (ctor_ != nullptr) {
963af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(ctor_); ctor_ != transformedNode) {
973af6ab5fSopenharmony_ci            ctor_->SetTransformedNode(transformationName, transformedNode);
983af6ab5fSopenharmony_ci            ctor_ = transformedNode->AsMethodDefinition();
993af6ab5fSopenharmony_ci        }
1003af6ab5fSopenharmony_ci    }
1013af6ab5fSopenharmony_ci
1023af6ab5fSopenharmony_ci    for (auto *&it : body_) {
1033af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(it); it != transformedNode) {
1043af6ab5fSopenharmony_ci            it->SetTransformedNode(transformationName, transformedNode);
1053af6ab5fSopenharmony_ci            it = transformedNode;
1063af6ab5fSopenharmony_ci        }
1073af6ab5fSopenharmony_ci    }
1083af6ab5fSopenharmony_ci}
1093af6ab5fSopenharmony_ci
1103af6ab5fSopenharmony_civoid ClassDefinition::Iterate(const NodeTraverser &cb) const
1113af6ab5fSopenharmony_ci{
1123af6ab5fSopenharmony_ci    if (ident_ != nullptr) {
1133af6ab5fSopenharmony_ci        cb(ident_);
1143af6ab5fSopenharmony_ci    }
1153af6ab5fSopenharmony_ci
1163af6ab5fSopenharmony_ci    if (typeParams_ != nullptr) {
1173af6ab5fSopenharmony_ci        cb(typeParams_);
1183af6ab5fSopenharmony_ci    }
1193af6ab5fSopenharmony_ci
1203af6ab5fSopenharmony_ci    if (superClass_ != nullptr) {
1213af6ab5fSopenharmony_ci        cb(superClass_);
1223af6ab5fSopenharmony_ci    }
1233af6ab5fSopenharmony_ci
1243af6ab5fSopenharmony_ci    if (superTypeParams_ != nullptr) {
1253af6ab5fSopenharmony_ci        cb(superTypeParams_);
1263af6ab5fSopenharmony_ci    }
1273af6ab5fSopenharmony_ci
1283af6ab5fSopenharmony_ci    // Survives adding new elements to the end
1293af6ab5fSopenharmony_ci    // NOLINTNEXTLINE(modernize-loop-convert)
1303af6ab5fSopenharmony_ci    for (size_t ix = 0; ix < implements_.size(); ix++) {
1313af6ab5fSopenharmony_ci        cb(implements_[ix]);
1323af6ab5fSopenharmony_ci    }
1333af6ab5fSopenharmony_ci
1343af6ab5fSopenharmony_ci    if (ctor_ != nullptr) {
1353af6ab5fSopenharmony_ci        cb(ctor_);
1363af6ab5fSopenharmony_ci    }
1373af6ab5fSopenharmony_ci
1383af6ab5fSopenharmony_ci    // NOLINTNEXTLINE(modernize-loop-convert)
1393af6ab5fSopenharmony_ci    for (size_t ix = 0; ix < body_.size(); ix++) {
1403af6ab5fSopenharmony_ci        cb(body_[ix]);
1413af6ab5fSopenharmony_ci    }
1423af6ab5fSopenharmony_ci}
1433af6ab5fSopenharmony_ci
1443af6ab5fSopenharmony_civoid ClassDefinition::SetIdent(ir::Identifier *ident) noexcept
1453af6ab5fSopenharmony_ci{
1463af6ab5fSopenharmony_ci    ident_ = ident;
1473af6ab5fSopenharmony_ci    if (ident_ != nullptr) {
1483af6ab5fSopenharmony_ci        ident_->SetParent(this);
1493af6ab5fSopenharmony_ci    }
1503af6ab5fSopenharmony_ci}
1513af6ab5fSopenharmony_ci
1523af6ab5fSopenharmony_civoid ClassDefinition::Dump(ir::AstDumper *dumper) const
1533af6ab5fSopenharmony_ci{
1543af6ab5fSopenharmony_ci    auto propFilter = [](AstNode *prop) -> bool {
1553af6ab5fSopenharmony_ci        return !prop->IsClassStaticBlock() || !prop->AsClassStaticBlock()->Function()->IsHidden();
1563af6ab5fSopenharmony_ci    };
1573af6ab5fSopenharmony_ci    dumper->Add({{"id", AstDumper::Nullish(ident_)},
1583af6ab5fSopenharmony_ci                 {"typeParameters", AstDumper::Optional(typeParams_)},
1593af6ab5fSopenharmony_ci                 {"superClass", AstDumper::Nullish(superClass_)},
1603af6ab5fSopenharmony_ci                 {"superTypeParameters", AstDumper::Optional(superTypeParams_)},
1613af6ab5fSopenharmony_ci                 {"implements", implements_},
1623af6ab5fSopenharmony_ci                 {"constructor", AstDumper::Optional(ctor_)},
1633af6ab5fSopenharmony_ci                 {"body", body_, propFilter}});
1643af6ab5fSopenharmony_ci}
1653af6ab5fSopenharmony_ci
1663af6ab5fSopenharmony_ci// This method is needed by OHOS CI code checker
1673af6ab5fSopenharmony_civoid ClassDefinition::DumpBody(ir::SrcDumper *dumper) const
1683af6ab5fSopenharmony_ci{
1693af6ab5fSopenharmony_ci    dumper->Add(" {");
1703af6ab5fSopenharmony_ci    if (!body_.empty()) {
1713af6ab5fSopenharmony_ci        dumper->IncrIndent();
1723af6ab5fSopenharmony_ci        dumper->Endl();
1733af6ab5fSopenharmony_ci        for (auto elem : body_) {
1743af6ab5fSopenharmony_ci            elem->Dump(dumper);
1753af6ab5fSopenharmony_ci            if (elem == body_.back()) {
1763af6ab5fSopenharmony_ci                dumper->DecrIndent();
1773af6ab5fSopenharmony_ci            }
1783af6ab5fSopenharmony_ci            dumper->Endl();
1793af6ab5fSopenharmony_ci        }
1803af6ab5fSopenharmony_ci    }
1813af6ab5fSopenharmony_ci    dumper->Add("}");
1823af6ab5fSopenharmony_ci}
1833af6ab5fSopenharmony_ci
1843af6ab5fSopenharmony_civoid ClassDefinition::Dump(ir::SrcDumper *dumper) const
1853af6ab5fSopenharmony_ci{
1863af6ab5fSopenharmony_ci    ASSERT(ident_ != nullptr);
1873af6ab5fSopenharmony_ci
1883af6ab5fSopenharmony_ci    if (IsExtern()) {
1893af6ab5fSopenharmony_ci        dumper->Add("extern ");
1903af6ab5fSopenharmony_ci    }
1913af6ab5fSopenharmony_ci
1923af6ab5fSopenharmony_ci    if (IsExported()) {
1933af6ab5fSopenharmony_ci        dumper->Add("export ");
1943af6ab5fSopenharmony_ci    }
1953af6ab5fSopenharmony_ci
1963af6ab5fSopenharmony_ci    if (IsDeclare()) {
1973af6ab5fSopenharmony_ci        dumper->Add("declare ");
1983af6ab5fSopenharmony_ci    }
1993af6ab5fSopenharmony_ci
2003af6ab5fSopenharmony_ci    if (IsFinal()) {
2013af6ab5fSopenharmony_ci        dumper->Add("final ");
2023af6ab5fSopenharmony_ci    }
2033af6ab5fSopenharmony_ci
2043af6ab5fSopenharmony_ci    if (IsAbstract()) {
2053af6ab5fSopenharmony_ci        dumper->Add("abstract ");
2063af6ab5fSopenharmony_ci    }
2073af6ab5fSopenharmony_ci
2083af6ab5fSopenharmony_ci    dumper->Add("class ");
2093af6ab5fSopenharmony_ci    ident_->Dump(dumper);
2103af6ab5fSopenharmony_ci
2113af6ab5fSopenharmony_ci    if (typeParams_ != nullptr) {
2123af6ab5fSopenharmony_ci        dumper->Add("<");
2133af6ab5fSopenharmony_ci        typeParams_->Dump(dumper);
2143af6ab5fSopenharmony_ci        dumper->Add("> ");
2153af6ab5fSopenharmony_ci    }
2163af6ab5fSopenharmony_ci
2173af6ab5fSopenharmony_ci    if (superClass_ != nullptr) {
2183af6ab5fSopenharmony_ci        dumper->Add(" extends ");
2193af6ab5fSopenharmony_ci        superClass_->Dump(dumper);
2203af6ab5fSopenharmony_ci    }
2213af6ab5fSopenharmony_ci
2223af6ab5fSopenharmony_ci    if (!implements_.empty()) {
2233af6ab5fSopenharmony_ci        dumper->Add(" implements ");
2243af6ab5fSopenharmony_ci        for (auto interface : implements_) {
2253af6ab5fSopenharmony_ci            interface->Dump(dumper);
2263af6ab5fSopenharmony_ci            if (interface != implements_.back()) {
2273af6ab5fSopenharmony_ci                dumper->Add(", ");
2283af6ab5fSopenharmony_ci            }
2293af6ab5fSopenharmony_ci        }
2303af6ab5fSopenharmony_ci    }
2313af6ab5fSopenharmony_ci
2323af6ab5fSopenharmony_ci    if (!IsDeclare() || !body_.empty()) {
2333af6ab5fSopenharmony_ci        DumpBody(dumper);
2343af6ab5fSopenharmony_ci    }
2353af6ab5fSopenharmony_ci    if (IsLocal()) {
2363af6ab5fSopenharmony_ci        dumper->Add(";");
2373af6ab5fSopenharmony_ci    }
2383af6ab5fSopenharmony_ci    dumper->Endl();
2393af6ab5fSopenharmony_ci}
2403af6ab5fSopenharmony_ci
2413af6ab5fSopenharmony_civoid ClassDefinition::Compile(compiler::PandaGen *pg) const
2423af6ab5fSopenharmony_ci{
2433af6ab5fSopenharmony_ci    pg->GetAstCompiler()->Compile(this);
2443af6ab5fSopenharmony_ci}
2453af6ab5fSopenharmony_ci
2463af6ab5fSopenharmony_civoid ClassDefinition::Compile(compiler::ETSGen *etsg) const
2473af6ab5fSopenharmony_ci{
2483af6ab5fSopenharmony_ci    etsg->GetAstCompiler()->Compile(this);
2493af6ab5fSopenharmony_ci}
2503af6ab5fSopenharmony_ci
2513af6ab5fSopenharmony_cichecker::Type *ClassDefinition::Check(checker::TSChecker *checker)
2523af6ab5fSopenharmony_ci{
2533af6ab5fSopenharmony_ci    return checker->GetAnalyzer()->Check(this);
2543af6ab5fSopenharmony_ci}
2553af6ab5fSopenharmony_ci
2563af6ab5fSopenharmony_cichecker::Type *ClassDefinition::Check(checker::ETSChecker *checker)
2573af6ab5fSopenharmony_ci{
2583af6ab5fSopenharmony_ci    return checker->GetAnalyzer()->Check(this);
2593af6ab5fSopenharmony_ci}
2603af6ab5fSopenharmony_ci
2613af6ab5fSopenharmony_ciint ClassDefinition::classCounter_ = 0;
2623af6ab5fSopenharmony_ci
2633af6ab5fSopenharmony_ci}  // namespace ark::es2panda::ir
264