#ifndef SPROUT_ITERATOR_SQUARE_ITERATOR_HPP #define SPROUT_ITERATOR_SQUARE_ITERATOR_HPP #include #include #include #include #include #include #include #include #include #include namespace sprout { // // square_iterator // template class square_iterator : public std::iterator< std::random_access_iterator_tag, Value, std::ptrdiff_t, Value*, Value > { public: typedef std::random_access_iterator_tag iterator_category; typedef Value value_type; typedef std::ptrdiff_t difference_type; typedef value_type* pointer; typedef value_type reference; private: difference_type index_; value_type frequency_; value_type amplitude_; value_type phase_; value_type duty_; private: explicit SPROUT_CONSTEXPR square_iterator(square_iterator const& other, difference_type index) : index_(index) , frequency_(other.frequency_) , amplitude_(other.amplitude_) , phase_(other.phase_) , duty_(other.duty_) {} public: SPROUT_CONSTEXPR square_iterator() : index_() , frequency_(1) , amplitude_(1) , phase_(0) , duty_(0.5) {} square_iterator(square_iterator const&) = default; explicit SPROUT_CONSTEXPR square_iterator( difference_type index, value_type const& frequency = 1, value_type const& amplitude = 1, value_type const& phase = 0, value_type const& duty = 0.5 ) : index_(index) , frequency_(frequency) , amplitude_(amplitude) , phase_(phase) , duty_(duty) {} template SPROUT_CONSTEXPR square_iterator(square_iterator const& it) : index_(it.index_) , frequency_(it.frequency_) , amplitude_(it.amplitude_) , phase_(it.phase_) , duty_(it.duty_) {} template square_iterator& operator=(square_iterator const& it) { square_iterator temp(it); temp.swap(*this); return *this; } SPROUT_CONSTEXPR difference_type index() const { return index_; } SPROUT_CONSTEXPR value_type const& frequency() const { return frequency_; } SPROUT_CONSTEXPR value_type const& amplitude() const { return amplitude_; } SPROUT_CONSTEXPR value_type const& phase() const { return phase_; } SPROUT_CONSTEXPR value_type const& duty() const { return duty_; } SPROUT_CONSTEXPR reference operator*() const { return amplitude_ == 0 ? 0 : amplitude_ * sprout::fixed::detail::square_value(frequency_ * value_type(index_) + phase_, duty_) ; } SPROUT_CONSTEXPR pointer operator->() const { return &operator*()(); } square_iterator& operator++() { ++index_; return *this; } square_iterator operator++(int) { square_iterator result(*this); ++index_; return result; } square_iterator& operator--() { --index_; return *this; } square_iterator operator--(int) { square_iterator temp(*this); --index_; return temp; } SPROUT_CONSTEXPR square_iterator operator+(difference_type n) const { return square_iterator(*this, index_ + n); } SPROUT_CONSTEXPR square_iterator operator-(difference_type n) const { return square_iterator(*this, index_ - n); } square_iterator& operator+=(difference_type n) { square_iterator temp(*this, index_ + n); temp.swap(*this); return *this; } square_iterator& operator-=(difference_type n) { square_iterator temp(*this, index_ - n); temp.swap(*this); return *this; } SPROUT_CONSTEXPR reference operator[](difference_type n) const { return amplitude_ == 0 ? 0 : amplitude_ * sprout::fixed::detail::square_value(frequency_ * value_type(index_ + n) + phase_, duty_) ; } SPROUT_CONSTEXPR square_iterator next() const { return square_iterator(*this, index_ + 1); } SPROUT_CONSTEXPR square_iterator prev() const { return square_iterator(*this, index_ - 1); } void swap(square_iterator& other) SPROUT_NOEXCEPT { sprout::swap(index_, other.index_); sprout::swap(frequency_, other.frequency_); sprout::swap(amplitude_, other.amplitude_); sprout::swap(phase_, other.phase_); sprout::swap(duty_, other.duty_); } }; template inline SPROUT_CONSTEXPR bool operator==(sprout::square_iterator const& lhs, sprout::square_iterator const& rhs) { return lhs.index() == rhs.index(); } template inline SPROUT_CONSTEXPR bool operator!=(sprout::square_iterator const& lhs, sprout::square_iterator const& rhs) { return !(lhs == rhs); } template inline SPROUT_CONSTEXPR bool operator<(sprout::square_iterator const& lhs, sprout::square_iterator const& rhs) { return lhs.index() < rhs.index(); } template inline SPROUT_CONSTEXPR bool operator>(sprout::square_iterator const& lhs, sprout::square_iterator const& rhs) { return rhs < lhs; } template inline SPROUT_CONSTEXPR bool operator<=(sprout::square_iterator const& lhs, sprout::square_iterator const& rhs) { return !(rhs < lhs); } template inline SPROUT_CONSTEXPR bool operator>=(sprout::square_iterator const& lhs, sprout::square_iterator const& rhs) { return !(lhs < rhs); } template inline SPROUT_CONSTEXPR typename sprout::square_iterator::difference_type operator-(sprout::square_iterator const& lhs, sprout::square_iterator const& rhs) { return lhs.index() - rhs.index(); } template inline SPROUT_CONSTEXPR sprout::square_iterator operator+(typename sprout::square_iterator::difference_type n, sprout::square_iterator const& it) { return it + n; } // // swap // template inline void swap(sprout::square_iterator& lhs, sprout::square_iterator& rhs) SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(lhs.swap(rhs))) { lhs.swap(rhs); } // // iterator_distance // template inline SPROUT_CONSTEXPR typename std::iterator_traits >::difference_type iterator_distance(sprout::square_iterator first, sprout::square_iterator last) { return last - first; } // // iterator_next // template inline SPROUT_CONSTEXPR sprout::square_iterator iterator_next(sprout::square_iterator const& it) { return it.next(); } template inline SPROUT_CONSTEXPR sprout::square_iterator iterator_next(sprout::square_iterator const& it, typename sprout::square_iterator::difference_type n) { return it + n; } // // iterator_prev // template inline SPROUT_CONSTEXPR sprout::square_iterator iterator_prev(sprout::square_iterator const& it) { return it.prev(); } template inline SPROUT_CONSTEXPR sprout::square_iterator iterator_prev(sprout::square_iterator const& it, typename sprout::square_iterator::difference_type n) { return it - n; } } // namespace sprout #endif // #ifndef SPROUT_ITERATOR_SQUARE_ITERATOR_HPP