// Copyright 2020 The Tint Authors.
//
// 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 SRC_AST_SWITCH_STATEMENT_H_
#define SRC_AST_SWITCH_STATEMENT_H_

#include <memory>
#include <utility>
#include <vector>

#include "src/ast/expression.h"
#include "src/ast/literal.h"
#include "src/ast/statement.h"
#include "src/ast/statement_condition.h"

namespace tint {
namespace ast {

/// A switch statement
class SwitchStatement : public Statement {
 public:
  /// Constructor
  SwitchStatement();
  /// Constructor
  /// @param condition the switch condition
  /// @param body the switch body
  SwitchStatement(std::unique_ptr<Expression> condition,
                  std::vector<std::unique_ptr<CaseStatement>> body);
  /// Constructor
  /// @param source the source information
  /// @param condition the switch condition
  /// @param body the switch body
  SwitchStatement(const Source& source,
                  std::unique_ptr<Expression> condition,
                  std::vector<std::unique_ptr<CaseStatement>> body);
  /// Move constructor
  SwitchStatement(SwitchStatement&&) = default;
  ~SwitchStatement() override;

  /// Sets the condition for the switch statement
  /// @param condition the condition to set
  void set_condition(std::unique_ptr<Expression> condition) {
    condition_ = std::move(condition);
  }
  /// @returns the switch condition or nullptr if none set
  Expression* condition() const { return condition_.get(); }
  /// @returns true if this is a default statement
  bool IsDefault() const { return condition_ == nullptr; }

  /// Sets the switch body
  /// @param body the switch body
  void set_body(std::vector<std::unique_ptr<CaseStatement>> body) {
    body_ = std::move(body);
  }
  /// @returns the Switch body
  const std::vector<std::unique_ptr<CaseStatement>>& body() const {
    return body_;
  }

  /// @returns true if this is a switch statement
  bool IsSwitch() const override { return true; }

  /// @returns true if the node is valid
  bool IsValid() const override;

  /// Writes a representation of the node to the output stream
  /// @param out the stream to write to
  /// @param indent number of spaces to indent the node when writing
  void to_str(std::ostream& out, size_t indent) const override;

 private:
  SwitchStatement(const SwitchStatement&) = delete;

  std::unique_ptr<Expression> condition_;
  std::vector<std::unique_ptr<CaseStatement>> body_;
};

}  // namespace ast
}  // namespace tint

#endif  // SRC_AST_SWITCH_STATEMENT_H_
