/**
 * 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_BINARY_EXPRESSION_H
#define ES2PANDA_IR_EXPRESSION_BINARY_EXPRESSION_H

#include "checker/checkerContext.h"
#include "compiler/core/vReg.h"
#include "ir/expression.h"

namespace ark::es2panda::checker {
class ETSAnalyzer;
}  // namespace ark::es2panda::checker

namespace ark::es2panda::ir {
class BinaryExpression : public Expression {
public:
    BinaryExpression() = delete;
    ~BinaryExpression() override = default;

    NO_COPY_SEMANTIC(BinaryExpression);
    NO_MOVE_SEMANTIC(BinaryExpression);

    explicit BinaryExpression(Expression *const left, Expression *const right, lexer::TokenType const operatorType)
        : Expression(AstNodeType::BINARY_EXPRESSION), left_(left), right_(right), operator_(operatorType)
    {
    }

    // NOTE (csabahurton): friend relationship can be removed once there are getters for private fields
    friend class checker::ETSAnalyzer;

    [[nodiscard]] const Expression *Left() const noexcept
    {
        return left_;
    }

    [[nodiscard]] Expression *Left() noexcept
    {
        return left_;
    }

    [[nodiscard]] const Expression *Right() const noexcept
    {
        return right_;
    }

    [[nodiscard]] Expression *Right() noexcept
    {
        return right_;
    }

    [[nodiscard]] const Expression *Result() const noexcept
    {
        return result_;
    }

    [[nodiscard]] Expression *Result() noexcept
    {
        return result_;
    }

    [[nodiscard]] lexer::TokenType OperatorType() const noexcept
    {
        return operator_;
    }

    [[nodiscard]] bool IsLogical() const noexcept
    {
        return operator_ == lexer::TokenType::PUNCTUATOR_NULLISH_COALESCING ||
               operator_ == lexer::TokenType::PUNCTUATOR_LOGICAL_OR ||
               operator_ == lexer::TokenType::PUNCTUATOR_LOGICAL_AND;
    }

    [[nodiscard]] bool IsLogicalExtended() const noexcept
    {
        return operator_ == lexer::TokenType::PUNCTUATOR_LOGICAL_AND ||
               operator_ == lexer::TokenType::PUNCTUATOR_LOGICAL_OR;
    }

    [[nodiscard]] bool IsBitwise() const noexcept
    {
        return operator_ == lexer::TokenType::PUNCTUATOR_BITWISE_OR ||
               operator_ == lexer::TokenType::PUNCTUATOR_BITWISE_XOR ||
               operator_ == lexer::TokenType::PUNCTUATOR_BITWISE_AND ||
               operator_ == lexer::TokenType::PUNCTUATOR_BITWISE_AND_EQUAL ||
               operator_ == lexer::TokenType::PUNCTUATOR_BITWISE_OR_EQUAL ||
               operator_ == lexer::TokenType::PUNCTUATOR_BITWISE_XOR_EQUAL;
    }

    [[nodiscard]] bool IsEquality() const noexcept
    {
        return operator_ == lexer::TokenType::PUNCTUATOR_EQUAL || operator_ == lexer::TokenType::PUNCTUATOR_NOT_EQUAL ||
               operator_ == lexer::TokenType::PUNCTUATOR_STRICT_EQUAL ||
               operator_ == lexer::TokenType::PUNCTUATOR_NOT_STRICT_EQUAL;
    }

    [[nodiscard]] bool IsArithmetic() const noexcept
    {
        return operator_ == lexer::TokenType::PUNCTUATOR_PLUS || operator_ == lexer::TokenType::PUNCTUATOR_MINUS ||
               operator_ == lexer::TokenType::PUNCTUATOR_MULTIPLY || operator_ == lexer::TokenType::PUNCTUATOR_DIVIDE ||
               operator_ == lexer::TokenType::PUNCTUATOR_MOD || operator_ == lexer::TokenType::PUNCTUATOR_PLUS_EQUAL ||
               operator_ == lexer::TokenType::PUNCTUATOR_MINUS_EQUAL ||
               operator_ == lexer::TokenType::PUNCTUATOR_MULTIPLY_EQUAL ||
               operator_ == lexer::TokenType::PUNCTUATOR_DIVIDE_EQUAL ||
               operator_ == lexer::TokenType::PUNCTUATOR_MOD_EQUAL || IsBitwise();
    }

    void SetLeft(Expression *expr) noexcept
    {
        left_ = expr;
        ES2PANDA_ASSERT(left_);
        left_->SetParent(this);
    }

    void SetRight(Expression *expr) noexcept
    {
        right_ = expr;
        ES2PANDA_ASSERT(right_);
        right_->SetParent(this);
    }

    void SetResult(Expression *expr) noexcept
    {
        result_ = expr;
        result_->SetParent(this);
    }

    void SetOperator(lexer::TokenType operatorType) noexcept
    {
        operator_ = operatorType;
        SetType(AstNodeType::BINARY_EXPRESSION);
    }

    [[nodiscard]] checker::Type *OperationType() noexcept
    {
        return operationType_;
    }

    void SetOperationType(checker::Type *const operationType) noexcept
    {
        operationType_ = operationType;
    }

    [[nodiscard]] const checker::Type *OperationType() const noexcept
    {
        return operationType_;
    }

    [[nodiscard]] BinaryExpression *Clone(ArenaAllocator *allocator, AstNode *parent) override;

    void TransformChildren(const NodeTransformer &cb, std::string_view transformationName) override;
    void Iterate(const NodeTraverser &cb) const override;
    void Dump(ir::AstDumper *dumper) const override;
    void Dump(ir::SrcDumper *dumper) const override;
    void Compile(compiler::PandaGen *pg) const override;
    void Compile(compiler::ETSGen *etsg) const override;
    void CompileOperands(compiler::ETSGen *etsg, compiler::VReg lhs) const;
    checker::Type *Check(checker::TSChecker *checker) override;
    checker::VerifiedType Check(checker::ETSChecker *checker) override;

    void Accept(ASTVisitorT *v) override
    {
        v->Accept(this);
    }

    void CleanUp() override
    {
        AstNode::CleanUp();
        operationType_ = nullptr;
    }

    void CleanCheckInformation() override;

private:
    EPtr<Expression> left_ = nullptr;
    EPtr<Expression> right_ = nullptr;
    EPtr<Expression> result_ = nullptr;
    lexer::TokenType operator_;
    EPtr<checker::Type> operationType_ {};
};
}  // namespace ark::es2panda::ir

#endif