243 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			243 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
//  (C) Copyright Joel de Guzman 2003.
 | 
						|
//  Distributed under the Boost Software License, Version 1.0. (See
 | 
						|
//  accompanying file LICENSE_1_0.txt or copy at
 | 
						|
//  http://www.boost.org/LICENSE_1_0.txt)
 | 
						|
 | 
						|
#ifndef VECTOR_INDEXING_SUITE_JDG20036_HPP
 | 
						|
# define VECTOR_INDEXING_SUITE_JDG20036_HPP
 | 
						|
 | 
						|
# include <boost/python/suite/indexing/indexing_suite.hpp>
 | 
						|
# include <boost/python/suite/indexing/container_utils.hpp>
 | 
						|
# include <boost/python/iterator.hpp>
 | 
						|
 | 
						|
namespace boost { namespace python {
 | 
						|
            
 | 
						|
    // Forward declaration
 | 
						|
    template <class Container, bool NoProxy, class DerivedPolicies>
 | 
						|
    class vector_indexing_suite;
 | 
						|
    
 | 
						|
    namespace detail
 | 
						|
    {
 | 
						|
        template <class Container, bool NoProxy>
 | 
						|
        class final_vector_derived_policies 
 | 
						|
            : public vector_indexing_suite<Container, 
 | 
						|
                NoProxy, final_vector_derived_policies<Container, NoProxy> > {};
 | 
						|
    }
 | 
						|
 | 
						|
    // The vector_indexing_suite class is a predefined indexing_suite derived 
 | 
						|
    // class for wrapping std::vector (and std::vector like) classes. It provides
 | 
						|
    // all the policies required by the indexing_suite (see indexing_suite).
 | 
						|
    // Example usage:
 | 
						|
    //
 | 
						|
    //  class X {...};
 | 
						|
    //
 | 
						|
    //  ...
 | 
						|
    //
 | 
						|
    //      class_<std::vector<X> >("XVec")
 | 
						|
    //          .def(vector_indexing_suite<std::vector<X> >())
 | 
						|
    //      ;
 | 
						|
    //
 | 
						|
    // By default indexed elements are returned by proxy. This can be
 | 
						|
    // disabled by supplying *true* in the NoProxy template parameter.
 | 
						|
    //
 | 
						|
    template <
 | 
						|
        class Container, 
 | 
						|
        bool NoProxy = false,
 | 
						|
        class DerivedPolicies 
 | 
						|
            = detail::final_vector_derived_policies<Container, NoProxy> >
 | 
						|
    class vector_indexing_suite 
 | 
						|
        : public indexing_suite<Container, DerivedPolicies, NoProxy>
 | 
						|
    {
 | 
						|
    public:
 | 
						|
    
 | 
						|
        typedef typename Container::value_type data_type;
 | 
						|
        typedef typename Container::value_type key_type;
 | 
						|
        typedef typename Container::size_type index_type;
 | 
						|
        typedef typename Container::size_type size_type;
 | 
						|
        typedef typename Container::difference_type difference_type;
 | 
						|
        
 | 
						|
        template <class Class>
 | 
						|
        static void 
 | 
						|
        extension_def(Class& cl)
 | 
						|
        {
 | 
						|
            cl
 | 
						|
                .def("append", &base_append)
 | 
						|
                .def("extend", &base_extend)
 | 
						|
            ;
 | 
						|
        }
 | 
						|
        
 | 
						|
        static 
 | 
						|
        typename mpl::if_<
 | 
						|
            is_class<data_type>
 | 
						|
          , data_type&
 | 
						|
          , data_type
 | 
						|
        >::type
 | 
						|
        get_item(Container& container, index_type i)
 | 
						|
        { 
 | 
						|
            return container[i];
 | 
						|
        }
 | 
						|
 | 
						|
        static object 
 | 
						|
        get_slice(Container& container, index_type from, index_type to)
 | 
						|
        { 
 | 
						|
            if (from > to)
 | 
						|
                return object(Container());
 | 
						|
            return object(Container(container.begin()+from, container.begin()+to));
 | 
						|
        }
 | 
						|
 | 
						|
        static void 
 | 
						|
        set_item(Container& container, index_type i, data_type const& v)
 | 
						|
        { 
 | 
						|
            container[i] = v;
 | 
						|
        }
 | 
						|
 | 
						|
        static void 
 | 
						|
        set_slice(Container& container, index_type from, 
 | 
						|
            index_type to, data_type const& v)
 | 
						|
        { 
 | 
						|
            if (from > to) {
 | 
						|
                return;
 | 
						|
            }
 | 
						|
            else {
 | 
						|
                container.erase(container.begin()+from, container.begin()+to);
 | 
						|
                container.insert(container.begin()+from, v);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        template <class Iter>
 | 
						|
        static void 
 | 
						|
        set_slice(Container& container, index_type from, 
 | 
						|
            index_type to, Iter first, Iter last)
 | 
						|
        { 
 | 
						|
            if (from > to) {
 | 
						|
                container.insert(container.begin()+from, first, last);
 | 
						|
            }
 | 
						|
            else {
 | 
						|
                container.erase(container.begin()+from, container.begin()+to);
 | 
						|
                container.insert(container.begin()+from, first, last);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        static void 
 | 
						|
        delete_item(Container& container, index_type i)
 | 
						|
        { 
 | 
						|
            container.erase(container.begin()+i);
 | 
						|
        }
 | 
						|
        
 | 
						|
        static void 
 | 
						|
        delete_slice(Container& container, index_type from, index_type to)
 | 
						|
        { 
 | 
						|
            if (from > to) {
 | 
						|
                // A null-op.
 | 
						|
                return;
 | 
						|
            }
 | 
						|
            container.erase(container.begin()+from, container.begin()+to);
 | 
						|
        }
 | 
						|
        
 | 
						|
        static size_t
 | 
						|
        size(Container& container)
 | 
						|
        {
 | 
						|
            return container.size();
 | 
						|
        }
 | 
						|
        
 | 
						|
        static bool
 | 
						|
        contains(Container& container, key_type const& key)
 | 
						|
        {
 | 
						|
            return std::find(container.begin(), container.end(), key)
 | 
						|
                != container.end();
 | 
						|
        }
 | 
						|
        
 | 
						|
        static index_type
 | 
						|
        get_min_index(Container& /*container*/)
 | 
						|
        { 
 | 
						|
            return 0;
 | 
						|
        }
 | 
						|
 | 
						|
        static index_type
 | 
						|
        get_max_index(Container& container)
 | 
						|
        { 
 | 
						|
            return container.size();
 | 
						|
        }
 | 
						|
      
 | 
						|
        static bool 
 | 
						|
        compare_index(Container& /*container*/, index_type a, index_type b)
 | 
						|
        {
 | 
						|
            return a < b;
 | 
						|
        }
 | 
						|
        
 | 
						|
        static index_type
 | 
						|
        convert_index(Container& container, PyObject* i_)
 | 
						|
        { 
 | 
						|
            extract<long> i(i_);
 | 
						|
            if (i.check())
 | 
						|
            {
 | 
						|
                long index = i();
 | 
						|
                if (index < 0)
 | 
						|
                    index += DerivedPolicies::size(container);
 | 
						|
                if (index >= long(container.size()) || index < 0)
 | 
						|
                {
 | 
						|
                    PyErr_SetString(PyExc_IndexError, "Index out of range");
 | 
						|
                    throw_error_already_set();
 | 
						|
                }
 | 
						|
                return index;
 | 
						|
            }
 | 
						|
            
 | 
						|
            PyErr_SetString(PyExc_TypeError, "Invalid index type");
 | 
						|
            throw_error_already_set();
 | 
						|
            return index_type();
 | 
						|
        }
 | 
						|
      
 | 
						|
        static void 
 | 
						|
        append(Container& container, data_type const& v)
 | 
						|
        { 
 | 
						|
            container.push_back(v);
 | 
						|
        }
 | 
						|
        
 | 
						|
        template <class Iter>
 | 
						|
        static void 
 | 
						|
        extend(Container& container, Iter first, Iter last)
 | 
						|
        { 
 | 
						|
            container.insert(container.end(), first, last);
 | 
						|
        }
 | 
						|
        
 | 
						|
    private:
 | 
						|
    
 | 
						|
        static void
 | 
						|
        base_append(Container& container, object v)
 | 
						|
        {
 | 
						|
            extract<data_type&> elem(v);
 | 
						|
            // try if elem is an exact Data
 | 
						|
            if (elem.check())
 | 
						|
            {
 | 
						|
                DerivedPolicies::append(container, elem());
 | 
						|
            }
 | 
						|
            else
 | 
						|
            {
 | 
						|
                //  try to convert elem to data_type
 | 
						|
                extract<data_type> elem(v);
 | 
						|
                if (elem.check())
 | 
						|
                {
 | 
						|
                    DerivedPolicies::append(container, elem());
 | 
						|
                }
 | 
						|
                else
 | 
						|
                {
 | 
						|
                    PyErr_SetString(PyExc_TypeError, 
 | 
						|
                        "Attempting to append an invalid type");
 | 
						|
                    throw_error_already_set();
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
        
 | 
						|
        static void
 | 
						|
        base_extend(Container& container, object v)
 | 
						|
        {
 | 
						|
            std::vector<data_type> temp;
 | 
						|
            container_utils::extend_container(temp, v);
 | 
						|
            DerivedPolicies::extend(container, temp.begin(), temp.end());
 | 
						|
        }
 | 
						|
    };
 | 
						|
       
 | 
						|
}} // namespace boost::python 
 | 
						|
 | 
						|
#endif // VECTOR_INDEXING_SUITE_JDG20036_HPP
 |