1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
|
/*
This file is part of Android File Transfer For Linux.
Copyright (C) 2015-2020 Vladimir Menshakov
This library is free software; you can redistribute it and/or modify it
under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 2.1 of the License,
or (at your option) any later version.
This library is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with this library; if not, write to the Free Software Foundation,
Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#ifndef AFTL_CLI_MAKETUPLE_H
#define AFTL_CLI_MAKETUPLE_H
#include <sstream>
#include <functional>
#include <mtp/types.h>
#include <mtp/Demangle.h>
namespace cli
{
namespace impl
{
template<typename T>
struct ValueConverter;
template<typename T>
struct StringStreamConverter
{
static T Convert(const std::string &text)
{
std::stringstream ss(text);
T value;
ss >> std::noskipws >> value;
return value;
}
};
template<>
struct ValueConverter<mtp::u32> : StringStreamConverter<mtp::u32>
{ };
template<>
struct ValueConverter<std::string>
{
static std::string Convert(const std::string &text)
{ return text; }
};
template<typename IteratorType, typename ... Tail>
struct TupleBuilder
{
std::tuple<> Result;
TupleBuilder(IteratorType begin, IteratorType end) { }
};
template<typename IteratorType, typename First, typename ... Tail>
struct TupleBuilder<IteratorType, First, Tail...>
{
typedef typename std::decay<First>::type ValueType;
std::string _text;
TupleBuilder<IteratorType, Tail...> _next;
typedef std::tuple<ValueType, typename std::decay<Tail>::type ... > ResultType;
ResultType Result;
static IteratorType Next(IteratorType & begin, IteratorType end)
{
if (begin == end)
throw std::runtime_error("not enough arguments");
return begin++;
}
TupleBuilder(IteratorType begin, IteratorType end):
_text(*Next(begin, end)), _next(begin, end)
{ Result = std::tuple_cat(std::make_tuple(ValueConverter<ValueType>::Convert(_text)), _next.Result); }
};
}
template<typename IteratorType>
std::tuple<> MakeTuple(IteratorType, IteratorType)
{ return std::tuple<>(); }
template<typename IteratorType, typename ...FuncArgs>
auto MakeTuple(IteratorType begin, IteratorType end) -> typename impl::TupleBuilder<IteratorType, FuncArgs...>::ResultType
{
impl::TupleBuilder<IteratorType, FuncArgs...> b(begin, end);
return b.Result;
}
};
#endif
|