/**
 * Copyright (c) 2021-2026 Huawei Device Co., Ltd.
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#ifndef ES2PANDA_IR_EXPRESSION_IDENTIFIER_H
#define ES2PANDA_IR_EXPRESSION_IDENTIFIER_H

#include <ir/expression.h>
#include <util/ustring.h>

namespace panda::es2panda::compiler {
class PandaGen;
}  // namespace panda::es2panda::compiler

namespace panda::es2panda::binder {
class Variable;
}  // namespace panda::es2panda::binder

namespace panda::es2panda::ir {

enum class IdentifierFlags : uint8_t {
    NONE,
    OPTIONAL = 1 << 0,
    REFERENCE = 1 << 1,
    TDZ = 1 << 2,
};

DEFINE_BITOPS(IdentifierFlags)

class Identifier : public Expression {
public:
    explicit Identifier(util::StringView name)
        : Expression(AstNodeType::IDENTIFIER), name_(name)
    {
    }

    explicit Identifier(util::StringView name, Expression *typeAnnotation)
        : Expression(AstNodeType::IDENTIFIER),
          name_(name),
          typeAnnotation_(typeAnnotation)
    {
    }

    const Expression *TypeAnnotation() const
    {
        return typeAnnotation_;
    }

    Expression *TypeAnnotation()
    {
        return typeAnnotation_;
    }

    const util::StringView &Name() const
    {
        return name_;
    }

    void SetName(util::StringView name)
    {
        name_ = name;
    }

    bool IsOptional() const
    {
        return (flags_ & IdentifierFlags::OPTIONAL) != 0;
    }

    void SetOptional(bool optional)
    {
        if (optional) {
            flags_ |= IdentifierFlags::OPTIONAL;
        } else {
            flags_ &= ~IdentifierFlags::OPTIONAL;
        }
    }

    bool IsReference() const
    {
        return (flags_ & IdentifierFlags::REFERENCE) != 0;
    }

    void SetReference()
    {
        flags_ |= IdentifierFlags::REFERENCE;
    }

    const std::vector<binder::Variable *> &TSVariables() const
    {
        return tsVariables_;
    }

    void SetTSVariables(const std::vector<binder::Variable *> &tsVariables)
    {
        tsVariables_ = tsVariables;
    }

    bool IsTdz() const
    {
        return (flags_ & IdentifierFlags::TDZ) != 0;
    }

    void SetTdz()
    {
        flags_ |= IdentifierFlags::TDZ;
    }

    void SetTsTypeAnnotation(Expression *typeAnnotation) override
    {
        typeAnnotation_ = typeAnnotation;
    }

    void Iterate(const NodeTraverser &cb) const override;
    void Dump(ir::AstDumper *dumper) const override;
    void Compile(compiler::PandaGen *pg) const override;
    void UpdateSelf(const NodeUpdater &cb, [[maybe_unused]] binder::Binder *binder) override;

private:
    util::StringView name_;
    Expression *typeAnnotation_ {};
    IdentifierFlags flags_ {IdentifierFlags::NONE};
    std::vector<binder::Variable *> tsVariables_;
};

}  // namespace panda::es2panda::ir

#endif