/* Copyright (C) 2008 - 2018 by David White Part of the Battle for Wesnoth Project https://www.wesnoth.org/ This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY. See the COPYING file for more details. */ #pragma once #include "formula/variant_value.hpp" #include #include namespace wfl { class formula_callable; class variant_iterator; class variant { public: enum DECIMAL_VARIANT_TYPE { DECIMAL_VARIANT }; variant(); explicit variant(int n); variant(int n, DECIMAL_VARIANT_TYPE /*type*/); variant(double n, DECIMAL_VARIANT_TYPE /*type*/); explicit variant(const std::vector& array); explicit variant(const std::string& str); explicit variant(const std::map& map); variant(const variant& v) = default; variant(variant&& v) = default; template variant(std::shared_ptr callable) : value_(std::make_shared(callable)) { assert(value_.get()); } variant& operator=(const variant& v) = default; variant& operator=(variant&& v) = default; variant operator[](size_t n) const; variant operator[](const variant& v) const; size_t num_elements() const; bool is_empty() const; variant get_member(const std::string& name) const; /** Functions to test the type of the internal value. */ bool is_null() const { return type() == VARIANT_TYPE::TYPE_NULL; } bool is_int() const { return type() == VARIANT_TYPE::TYPE_INT; } bool is_decimal() const { return type() == VARIANT_TYPE::TYPE_DECIMAL; } bool is_callable() const { return type() == VARIANT_TYPE::TYPE_CALLABLE; } bool is_list() const { return type() == VARIANT_TYPE::TYPE_LIST; } bool is_string() const { return type() == VARIANT_TYPE::TYPE_STRING; } bool is_map() const { return type() == VARIANT_TYPE::TYPE_MAP; } int as_int() const; /** Returns variant's internal representation of decimal number: ie, 1.234 is represented as 1234 */ int as_decimal() const; /** Returns a boolean state of the variant value. The implementation is type-dependent. */ bool as_bool() const; const std::vector& as_list() const; const std::map& as_map() const; const std::string& as_string() const; const_formula_callable_ptr as_callable() const { must_be(VARIANT_TYPE::TYPE_CALLABLE); return value_cast()->get_callable(); } template std::shared_ptr try_convert() const { if(!is_callable()) { return nullptr; } return std::dynamic_pointer_cast(std::const_pointer_cast(as_callable())); } template std::shared_ptr convert_to() const { std::shared_ptr res = std::dynamic_pointer_cast(std::const_pointer_cast(as_callable())); if(!res) { throw type_error("could not convert type"); } return res; } variant operator+(const variant&) const; variant operator-(const variant&) const; variant operator*(const variant&) const; variant operator/(const variant&) const; variant operator^(const variant&) const; variant operator%(const variant&) const; variant operator-() const; bool operator==(const variant&) const; bool operator!=(const variant&) const; bool operator<(const variant&) const; bool operator>(const variant&) const; bool operator<=(const variant&) const; bool operator>=(const variant&) const; variant list_elements_add(const variant& v) const; variant list_elements_sub(const variant& v) const; variant list_elements_mul(const variant& v) const; variant list_elements_div(const variant& v) const; variant concatenate(const variant& v) const; variant build_range(const variant& v) const; bool contains(const variant& other) const; variant get_keys() const; variant get_values() const; variant_iterator begin() const; variant_iterator end() const; std::string serialize_to_string() const; void serialize_from_string(const std::string& str); std::string string_cast() const; std::string to_debug_string(bool verbose = false, formula_seen_stack* seen = nullptr) const; /** Gets string name of the current value type */ std::string type_string() const { return type().to_string(); } variant execute_variant(const variant& to_exec); private: template std::shared_ptr value_cast() const { return wfl::value_cast(value_); } void must_be(VARIANT_TYPE t) const; void must_both_be(VARIANT_TYPE t, const variant& second) const; VARIANT_TYPE type() const { return value_->get_type(); } /** * Variant value. * Each of the constructors initialized this with the appropriate helper class. */ value_base_ptr value_; }; /** * Iterator class for the variant. * * Depending on the @p type_ the @p list_iterator_ and the @p map_iterator_ are * a valid iterator or singular. Since most actions on singular iterators * result in Undefined Behavior care should be taken when copying the * @p list_iterator_ and @p map_iterator_. */ class variant_iterator { public: typedef variant value_type; typedef std::bidirectional_iterator_tag iterator_category; typedef variant& reference; typedef variant* pointer; typedef int difference_type; /** * Constructor for a no-op iterator. */ variant_iterator(); /** * Constructor for a generic iterator. * * @pre @p iter is not singular. * * @param value A pointer to a variant value representing the container. * @param iter An underlying iterator for the underlying container. */ variant_iterator(const variant_value_base* value, const boost::any& iter); variant operator*() const; variant_iterator& operator++(); variant_iterator operator++(int); variant_iterator& operator--(); variant_iterator operator--(int); bool operator==(const variant_iterator& that) const; bool operator!=(const variant_iterator& that) const; private: VARIANT_TYPE type_; const variant_value_base* container_; boost::any iter_; }; }