1 // (C) Copyright Joel de Guzman 2003.
2 // Distributed under the Boost Software License, Version 1.0. (See
3 // accompanying file LICENSE_1_0.txt or copy at
4 // http://www.boost.org/LICENSE_1_0.txt)
6 #ifndef VECTOR_INDEXING_SUITE_JDG20036_HPP
7 # define VECTOR_INDEXING_SUITE_JDG20036_HPP
9 # include <boost/python/suite/indexing/indexing_suite.hpp>
10 # include <boost/python/suite/indexing/container_utils.hpp>
11 # include <boost/python/iterator.hpp>
13 namespace boost { namespace python {
15 // Forward declaration
16 template <class Container, bool NoProxy, class DerivedPolicies>
17 class vector_indexing_suite;
21 template <class Container, bool NoProxy>
22 class final_vector_derived_policies
23 : public vector_indexing_suite<Container,
24 NoProxy, final_vector_derived_policies<Container, NoProxy> > {};
27 // The vector_indexing_suite class is a predefined indexing_suite derived
28 // class for wrapping std::vector (and std::vector like) classes. It provides
29 // all the policies required by the indexing_suite (see indexing_suite).
36 // class_<std::vector<X> >("XVec")
37 // .def(vector_indexing_suite<std::vector<X> >())
40 // By default indexed elements are returned by proxy. This can be
41 // disabled by supplying *true* in the NoProxy template parameter.
47 = detail::final_vector_derived_policies<Container, NoProxy> >
48 class vector_indexing_suite
49 : public indexing_suite<Container, DerivedPolicies, NoProxy>
53 typedef typename Container::value_type data_type;
54 typedef typename Container::value_type key_type;
55 typedef typename Container::size_type index_type;
56 typedef typename Container::size_type size_type;
57 typedef typename Container::difference_type difference_type;
59 template <class Class>
61 extension_def(Class& cl)
64 .def("append", &base_append)
65 .def("extend", &base_extend)
75 get_item(Container& container, index_type i)
81 get_slice(Container& container, index_type from, index_type to)
84 return object(Container());
85 return object(Container(container.begin()+from, container.begin()+to));
89 set_item(Container& container, index_type i, data_type const& v)
95 set_slice(Container& container, index_type from,
96 index_type to, data_type const& v)
102 container.erase(container.begin()+from, container.begin()+to);
103 container.insert(container.begin()+from, v);
107 template <class Iter>
109 set_slice(Container& container, index_type from,
110 index_type to, Iter first, Iter last)
113 container.insert(container.begin()+from, first, last);
116 container.erase(container.begin()+from, container.begin()+to);
117 container.insert(container.begin()+from, first, last);
122 delete_item(Container& container, index_type i)
124 container.erase(container.begin()+i);
128 delete_slice(Container& container, index_type from, index_type to)
134 container.erase(container.begin()+from, container.begin()+to);
138 size(Container& container)
140 return container.size();
144 contains(Container& container, key_type const& key)
146 return std::find(container.begin(), container.end(), key)
151 get_min_index(Container& /*container*/)
157 get_max_index(Container& container)
159 return container.size();
163 compare_index(Container& /*container*/, index_type a, index_type b)
169 convert_index(Container& container, PyObject* i_)
176 index += DerivedPolicies::size(container);
177 if (index >= long(container.size()) || index < 0)
179 PyErr_SetString(PyExc_IndexError, "Index out of range");
180 throw_error_already_set();
185 PyErr_SetString(PyExc_TypeError, "Invalid index type");
186 throw_error_already_set();
191 append(Container& container, data_type const& v)
193 container.push_back(v);
196 template <class Iter>
198 extend(Container& container, Iter first, Iter last)
200 container.insert(container.end(), first, last);
206 base_append(Container& container, object v)
208 extract<data_type&> elem(v);
209 // try if elem is an exact Data
212 DerivedPolicies::append(container, elem());
216 // try to convert elem to data_type
217 extract<data_type> elem(v);
220 DerivedPolicies::append(container, elem());
224 PyErr_SetString(PyExc_TypeError,
225 "Attempting to append an invalid type");
226 throw_error_already_set();
232 base_extend(Container& container, object v)
234 std::vector<data_type> temp;
235 container_utils::extend_container(temp, v);
236 DerivedPolicies::extend(container, temp.begin(), temp.end());
240 }} // namespace boost::python
242 #endif // VECTOR_INDEXING_SUITE_JDG20036_HPP