2013-06-29 16:05:48 +03:00
# pragma once
2012-03-13 15:47:47 +03:00
/*
* Global . h , part of VCMI engine
*
* Authors : listed in file AUTHORS in main folder
*
* License : GNU General Public License v2 .0 or later
* Full text of license available in license . txt file , in main folder
*
*/
2013-04-05 13:29:46 +03:00
/* ---------------------------------------------------------------------------- */
/* Compiler detection */
/* ---------------------------------------------------------------------------- */
// Fixed width bool data type is important for serialization
static_assert ( sizeof ( bool ) = = 1 , " Bool needs to be 1 byte in size. " ) ;
# if defined _M_X64 && defined _WIN32 //Win64 -> cannot load 32-bit DLLs for video handling
2013-06-29 16:05:48 +03:00
# define DISABLE_VIDEO
2013-04-05 13:29:46 +03:00
# endif
# ifdef __GNUC__
2013-06-29 16:05:48 +03:00
# define GCC_VERSION (__GNUC__ * 100 + __GNUC_MINOR__ * 10 + __GNUC_PATCHLEVEL__)
# endif
2014-05-29 13:42:05 +03:00
# if !defined(__clang__) && defined(__GNUC__) && (GCC_VERSION < 470)
# error VCMI requires at least gcc-4.7.2 for successful compilation or clang-3.1. Please update your compiler
2013-04-05 13:29:46 +03:00
# endif
# if defined(__GNUC__) && (GCC_VERSION == 470 || GCC_VERSION == 471)
2014-05-29 13:42:05 +03:00
# error This GCC version has buggy std::array::at version and should not be used. Please update to 4.7.2 or later
2013-04-05 13:29:46 +03:00
# endif
/* ---------------------------------------------------------------------------- */
/* Guarantee compiler features */
/* ---------------------------------------------------------------------------- */
//defining available c++11 features
2013-06-29 16:05:48 +03:00
//initialization lists - gcc or clang
2013-06-26 14:18:27 +03:00
# if defined(__clang__) || defined(__GNUC__)
2013-06-29 16:05:48 +03:00
# define CPP11_USE_INITIALIZERS_LIST
2013-04-05 13:29:46 +03:00
# endif
/* ---------------------------------------------------------------------------- */
/* Suppress some compiler warnings */
/* ---------------------------------------------------------------------------- */
2013-02-04 00:05:44 +03:00
# ifdef _MSC_VER
2013-06-29 16:05:48 +03:00
# pragma warning (disable : 4800 ) /* disable conversion to bool warning -- I think it's intended in all places */
2013-04-05 13:29:46 +03:00
# endif
2013-02-04 00:05:44 +03:00
2013-04-05 13:29:46 +03:00
/* ---------------------------------------------------------------------------- */
/* Commonly used C++, Boost headers */
/* ---------------------------------------------------------------------------- */
2012-03-13 15:47:47 +03:00
# define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers
2013-08-02 11:28:52 +03:00
# define _USE_MATH_DEFINES
2012-03-13 15:47:47 +03:00
# include <cstdio>
# include <stdio.h>
2012-08-08 11:25:27 +03:00
# include <algorithm>
# include <array>
2012-03-13 15:47:47 +03:00
# include <cassert>
2012-08-08 11:25:27 +03:00
# include <climits>
# include <cmath>
# include <cstdlib>
2013-06-26 14:18:27 +03:00
# include <functional>
2012-08-08 11:25:27 +03:00
# include <fstream>
# include <iomanip>
# include <iostream>
2012-03-13 15:47:47 +03:00
# include <map>
2012-08-08 11:25:27 +03:00
# include <memory>
# include <numeric>
2012-03-13 15:47:47 +03:00
# include <queue>
2013-06-26 14:18:27 +03:00
# include <random>
2012-03-13 15:47:47 +03:00
# include <set>
2012-08-08 11:25:27 +03:00
# include <sstream>
2012-08-26 12:07:48 +03:00
# include <string>
2013-06-29 16:05:48 +03:00
# include <unordered_set>
# include <unordered_map>
2012-03-13 15:47:47 +03:00
# include <utility>
2012-08-26 12:07:48 +03:00
# include <vector>
2012-03-13 15:47:47 +03:00
2012-07-02 12:04:05 +03:00
//The only available version is 3, as of Boost 1.50
2012-09-16 18:49:22 +03:00
# include <boost/version.hpp>
2012-07-02 12:04:05 +03:00
# define BOOST_FILESYSTEM_VERSION 3
2014-02-05 23:25:36 +03:00
# if BOOST_VERSION > 105000
2012-09-15 22:16:16 +03:00
# define BOOST_THREAD_VERSION 3
2012-09-16 18:49:22 +03:00
# endif
2012-09-15 22:16:16 +03:00
# define BOOST_THREAD_DONT_PROVIDE_THREAD_DESTRUCTOR_CALLS_TERMINATE_IF_JOINABLE 1
2013-07-02 18:23:32 +03:00
//#define BOOST_BIND_NO_PLACEHOLDERS
2012-03-13 15:47:47 +03:00
# include <boost/algorithm/string.hpp>
# include <boost/assign.hpp>
# include <boost/cstdint.hpp>
2013-04-10 19:28:14 +03:00
# include <boost/current_function.hpp>
2013-11-08 23:36:26 +03:00
# include <boost/crc.hpp>
2012-03-13 15:47:47 +03:00
# include <boost/date_time/posix_time/posix_time.hpp>
2013-04-05 13:29:46 +03:00
# include <boost/date_time/posix_time/posix_time_io.hpp>
2012-03-13 15:47:47 +03:00
# include <boost/filesystem.hpp>
# include <boost/format.hpp>
# include <boost/lexical_cast.hpp>
# include <boost/logic/tribool.hpp>
2012-08-26 12:07:48 +03:00
# include <boost/optional.hpp>
2013-06-26 14:18:27 +03:00
# include <boost/program_options.hpp>
2013-02-09 22:19:14 +03:00
# include <boost/range/adaptor/filtered.hpp>
2013-06-29 16:05:48 +03:00
# include <boost/range/adaptor/reversed.hpp>
2013-06-26 14:18:27 +03:00
# include <boost/range/algorithm.hpp>
2012-03-13 15:47:47 +03:00
# include <boost/thread.hpp>
2012-04-14 05:20:22 +03:00
# include <boost/variant.hpp>
2013-07-07 11:27:27 +03:00
# include <boost/math/special_functions/round.hpp>
2012-03-13 15:47:47 +03:00
# ifdef ANDROID
# include <android/log.h>
# endif
2013-04-05 13:29:46 +03:00
/* ---------------------------------------------------------------------------- */
/* Usings */
/* ---------------------------------------------------------------------------- */
using std : : shared_ptr ;
using std : : unique_ptr ;
using std : : make_shared ;
2013-07-02 18:23:32 +03:00
//using namespace std::placeholders;
2013-04-05 13:29:46 +03:00
namespace range = boost : : range ;
/* ---------------------------------------------------------------------------- */
/* Typedefs */
/* ---------------------------------------------------------------------------- */
2012-03-13 15:47:47 +03:00
// Integral data types
typedef boost : : uint64_t ui64 ; //unsigned int 64 bits (8 bytes)
typedef boost : : uint32_t ui32 ; //unsigned int 32 bits (4 bytes)
typedef boost : : uint16_t ui16 ; //unsigned int 16 bits (2 bytes)
typedef boost : : uint8_t ui8 ; //unsigned int 8 bits (1 byte)
typedef boost : : int64_t si64 ; //signed int 64 bits (8 bytes)
typedef boost : : int32_t si32 ; //signed int 32 bits (4 bytes)
typedef boost : : int16_t si16 ; //signed int 16 bits (2 bytes)
typedef boost : : int8_t si8 ; //signed int 8 bits (1 byte)
2013-04-05 13:29:46 +03:00
// Lock typedefs
typedef boost : : lock_guard < boost : : mutex > TLockGuard ;
2013-04-09 17:31:36 +03:00
typedef boost : : lock_guard < boost : : recursive_mutex > TLockGuardRec ;
2012-03-13 15:47:47 +03:00
2013-04-05 13:29:46 +03:00
/* ---------------------------------------------------------------------------- */
/* Macros */
/* ---------------------------------------------------------------------------- */
2012-03-13 15:47:47 +03:00
// Import + Export macro declarations
# ifdef _WIN32
2013-06-26 14:18:27 +03:00
# ifdef __GNUC__
# define DLL_EXPORT __attribute__((dllexport))
# else
# define DLL_EXPORT __declspec(dllexport)
# endif
2012-03-13 15:47:47 +03:00
# else
2013-06-26 14:18:27 +03:00
# ifdef __GNUC__
# define DLL_EXPORT __attribute__ ((visibility("default")))
# endif
2012-03-13 15:47:47 +03:00
# endif
# ifdef _WIN32
2013-06-26 14:18:27 +03:00
# ifdef __GNUC__
# define DLL_IMPORT __attribute__((dllimport))
# else
# define DLL_IMPORT __declspec(dllimport)
# endif
2012-09-15 22:16:16 +03:00
# else
2013-06-26 14:18:27 +03:00
# ifdef __GNUC__
# define DLL_IMPORT __attribute__ ((visibility("default")))
# endif
2012-03-13 15:47:47 +03:00
# endif
# ifdef VCMI_DLL
2013-06-26 14:18:27 +03:00
# define DLL_LINKAGE DLL_EXPORT
2012-03-13 15:47:47 +03:00
# else
2013-06-26 14:18:27 +03:00
# define DLL_LINKAGE DLL_IMPORT
2012-03-13 15:47:47 +03:00
# endif
2013-04-05 13:29:46 +03:00
# define THROW_FORMAT(message, formatting_elems) throw std::runtime_error(boost::str(boost::format(message) % formatting_elems))
2012-03-13 15:47:47 +03:00
2013-04-11 18:58:01 +03:00
# define ASSERT_IF_CALLED_WITH_PLAYER if(!player) {logGlobal->errorStream() << BOOST_CURRENT_FUNCTION; assert(0);}
2012-03-13 15:47:47 +03:00
2013-04-05 13:29:46 +03:00
//XXX pls dont - 'debug macros' are usually more trouble than it's worth
# define HANDLE_EXCEPTION \
catch ( const std : : exception & e ) { \
2013-04-11 18:58:01 +03:00
logGlobal - > errorStream ( ) < < e . what ( ) ; \
2013-04-05 13:29:46 +03:00
throw ; \
} \
catch ( const std : : exception * e ) \
{ \
2013-04-11 18:58:01 +03:00
logGlobal - > errorStream ( ) < < e - > what ( ) ; \
2013-04-05 13:29:46 +03:00
throw ; \
} \
catch ( const std : : string & e ) { \
2013-04-11 18:58:01 +03:00
logGlobal - > errorStream ( ) < < e ; \
2013-04-05 13:29:46 +03:00
throw ; \
}
2012-03-13 15:47:47 +03:00
2013-04-05 13:29:46 +03:00
# define HANDLE_EXCEPTIONC(COMMAND) \
catch ( const std : : exception & e ) { \
COMMAND ; \
2013-04-11 18:58:01 +03:00
logGlobal - > errorStream ( ) < < e . what ( ) ; \
2013-04-05 13:29:46 +03:00
throw ; \
} \
catch ( const std : : string & e ) \
{ \
COMMAND ; \
2013-04-11 18:58:01 +03:00
logGlobal - > errorStream ( ) < < e ; \
2013-04-05 13:29:46 +03:00
throw ; \
}
2012-03-13 15:47:47 +03:00
2013-04-05 13:29:46 +03:00
// can be used for counting arrays
template < typename T , size_t N > char ( & _ArrayCountObj ( const T ( & ) [ N ] ) ) [ N ] ;
# define ARRAY_COUNT(arr) (sizeof(_ArrayCountObj(arr)))
2012-03-13 15:47:47 +03:00
2013-11-09 16:49:36 +03:00
// should be used for variables that becomes unused in release builds (e.g. only used for assert checks)
# define UNUSED(VAR) ((void)VAR)
2013-12-03 18:24:13 +03:00
/* ---------------------------------------------------------------------------- */
/* VCMI standard library */
/* ---------------------------------------------------------------------------- */
2013-12-28 15:47:55 +03:00
# include "lib/logging/CLogger.h"
void inline handleException ( )
{
try
{
throw ;
}
catch ( const std : : exception & ex )
{
logGlobal - > errorStream ( ) < < ex . what ( ) ;
}
catch ( const std : : string & ex )
{
logGlobal - > errorStream ( ) < < ex ;
}
catch ( . . . )
{
logGlobal - > errorStream ( ) < < " Sorry, caught unknown exception type. No more info available. " ;
}
}
2013-12-03 18:24:13 +03:00
template < typename T >
2013-12-08 20:54:13 +03:00
std : : ostream & operator < < ( std : : ostream & out , const boost : : optional < T > & opt )
2013-12-03 18:24:13 +03:00
{
2013-12-08 20:54:13 +03:00
if ( opt ) return out < < * opt ;
else return out < < " empty " ;
2013-08-18 18:46:28 +03:00
}
2013-12-08 20:54:13 +03:00
template < typename T >
std : : ostream & operator < < ( std : : ostream & out , const std : : vector < T > & container )
{
out < < " [ " ;
for ( auto it = container . begin ( ) ; it ! = container . end ( ) ; + + it )
{
out < < * it ;
if ( std : : prev ( container . end ( ) ) ! = it ) out < < " , " ;
}
return out < < " ] " ;
}
2013-08-18 18:46:28 +03:00
2012-03-13 15:47:47 +03:00
namespace vstd
{
2013-06-29 16:05:48 +03:00
// combine hashes. Present in boost but not in std
template < class T >
inline void hash_combine ( std : : size_t & seed , const T & v )
{
std : : hash < T > hasher ;
seed ^ = hasher ( v ) + 0x9e3779b9 + ( seed < < 6 ) + ( seed > > 2 ) ;
}
2012-03-13 15:47:47 +03:00
//returns true if container c contains item i
template < typename Container , typename Item >
bool contains ( const Container & c , const Item & i )
{
2013-06-29 16:05:48 +03:00
return std : : find ( std : : begin ( c ) , std : : end ( c ) , i ) ! = std : : end ( c ) ;
2012-08-26 12:07:48 +03:00
}
//returns true if container c contains item i
template < typename Container , typename Pred >
bool contains_if ( const Container & c , Pred p )
{
2013-06-29 16:05:48 +03:00
return std : : find_if ( std : : begin ( c ) , std : : end ( c ) , p ) ! = std : : end ( c ) ;
2012-03-13 15:47:47 +03:00
}
//returns true if map c contains item i
template < typename V , typename Item , typename Item2 >
bool contains ( const std : : map < Item , V > & c , const Item2 & i )
{
2013-12-03 18:24:13 +03:00
return c . find ( i ) ! = c . end ( ) ;
}
//returns true if unordered set c contains item i
template < typename Item >
bool contains ( const std : : unordered_set < Item > & c , const Item & i )
2012-03-13 15:47:47 +03:00
{
return c . find ( i ) ! = c . end ( ) ;
}
2013-03-03 21:00:37 +03:00
template < typename V , typename Item , typename Item2 >
2013-06-29 16:05:48 +03:00
bool contains ( const std : : unordered_map < Item , V > & c , const Item2 & i )
2013-03-03 21:00:37 +03:00
{
return c . find ( i ) ! = c . end ( ) ;
}
2012-03-13 15:47:47 +03:00
//returns position of first element in vector c equal to s, if there is no such element, -1 is returned
2012-06-13 16:04:06 +03:00
template < typename Container , typename T2 >
int find_pos ( const Container & c , const T2 & s )
2012-03-13 15:47:47 +03:00
{
2012-06-13 16:04:06 +03:00
size_t i = 0 ;
2013-06-29 16:05:48 +03:00
for ( auto iter = std : : begin ( c ) ; iter ! = std : : end ( c ) ; iter + + , i + + )
2012-06-13 16:04:06 +03:00
if ( * iter = = s )
2012-03-13 15:47:47 +03:00
return i ;
return - 1 ;
}
2013-07-22 01:01:29 +03:00
//Func f tells if element matches
template < typename Container , typename Func >
int find_pos_if ( const Container & c , const Func & f )
2012-03-13 15:47:47 +03:00
{
2013-07-22 01:01:29 +03:00
auto ret = boost : : range : : find_if ( c , f ) ;
if ( ret ! = std : : end ( c ) )
return std : : distance ( std : : begin ( c ) , ret ) ;
2012-03-13 15:47:47 +03:00
return - 1 ;
}
//returns iterator to the given element if present in container, end() if not
template < typename Container , typename Item >
typename Container : : iterator find ( Container & c , const Item & i )
{
return std : : find ( c . begin ( ) , c . end ( ) , i ) ;
}
//returns const iterator to the given element if present in container, end() if not
template < typename Container , typename Item >
typename Container : : const_iterator find ( const Container & c , const Item & i )
{
return std : : find ( c . begin ( ) , c . end ( ) , i ) ;
}
//removes element i from container c, returns false if c does not contain i
template < typename Container , typename Item >
typename Container : : size_type operator - = ( Container & c , const Item & i )
{
typename Container : : iterator itr = find ( c , i ) ;
if ( itr = = c . end ( ) )
return false ;
c . erase ( itr ) ;
return true ;
}
//assigns greater of (a, b) to a and returns maximum of (a, b)
template < typename t1 , typename t2 >
t1 & amax ( t1 & a , const t2 & b )
{
if ( a > = b )
return a ;
else
{
a = b ;
return a ;
}
}
//assigns smaller of (a, b) to a and returns minimum of (a, b)
template < typename t1 , typename t2 >
t1 & amin ( t1 & a , const t2 & b )
{
if ( a < = b )
return a ;
else
{
a = b ;
return a ;
}
}
//makes a to fit the range <b, c>
template < typename t1 , typename t2 , typename t3 >
t1 & abetween ( t1 & a , const t2 & b , const t3 & c )
{
amax ( a , b ) ;
amin ( a , c ) ;
return a ;
}
//checks if a is between b and c
template < typename t1 , typename t2 , typename t3 >
2012-08-06 10:34:37 +03:00
bool isbetween ( const t1 & value , const t2 & min , const t3 & max )
2012-03-13 15:47:47 +03:00
{
2012-08-06 10:34:37 +03:00
return value > min & & value < max ;
2012-03-13 15:47:47 +03:00
}
//checks if a is within b and c
template < typename t1 , typename t2 , typename t3 >
2012-08-06 10:34:37 +03:00
bool iswithin ( const t1 & value , const t2 & min , const t3 & max )
2012-03-13 15:47:47 +03:00
{
2012-08-06 10:34:37 +03:00
return value > = min & & value < = max ;
2012-03-13 15:47:47 +03:00
}
template < typename t1 , typename t2 >
struct assigner
{
public :
t1 & op1 ;
t2 op2 ;
assigner ( t1 & a1 , const t2 & a2 )
: op1 ( a1 ) , op2 ( a2 )
{ }
void operator ( ) ( )
{
op1 = op2 ;
}
} ;
// Assigns value a2 to a1. The point of time of the real operation can be controlled
// with the () operator.
template < typename t1 , typename t2 >
assigner < t1 , t2 > assigno ( t1 & a1 , const t2 & a2 )
{
return assigner < t1 , t2 > ( a1 , a2 ) ;
}
2013-06-26 14:18:27 +03:00
//deleted pointer and sets it to nullptr
2012-03-13 15:47:47 +03:00
template < typename T >
void clear_pointer ( T * & ptr )
{
delete ptr ;
2013-06-26 14:18:27 +03:00
ptr = nullptr ;
2012-03-13 15:47:47 +03:00
}
2013-07-02 15:08:30 +03:00
# if _MSC_VER >= 1800
using std : : make_unique ;
# else
2012-02-20 00:03:43 +03:00
template < typename T >
std : : unique_ptr < T > make_unique ( )
{
return std : : unique_ptr < T > ( new T ( ) ) ;
2012-03-13 15:47:47 +03:00
}
2012-02-20 00:03:43 +03:00
template < typename T , typename Arg1 >
2012-04-08 04:15:18 +03:00
std : : unique_ptr < T > make_unique ( Arg1 & & arg1 )
2012-02-20 00:03:43 +03:00
{
return std : : unique_ptr < T > ( new T ( std : : forward < Arg1 > ( arg1 ) ) ) ;
2012-03-13 15:47:47 +03:00
}
2012-04-08 04:15:18 +03:00
template < typename T , typename Arg1 , typename Arg2 >
std : : unique_ptr < T > make_unique ( Arg1 & & arg1 , Arg2 & & arg2 )
{
return std : : unique_ptr < T > ( new T ( std : : forward < Arg1 > ( arg1 ) , std : : forward < Arg2 > ( arg2 ) ) ) ;
}
2013-01-20 17:43:58 +03:00
template < typename T , typename Arg1 , typename Arg2 , typename Arg3 >
std : : unique_ptr < T > make_unique ( Arg1 & & arg1 , Arg2 & & arg2 , Arg3 & & arg3 )
{
return std : : unique_ptr < T > ( new T ( std : : forward < Arg1 > ( arg1 ) , std : : forward < Arg2 > ( arg2 ) , std : : forward < Arg3 > ( arg3 ) ) ) ;
}
2013-04-19 14:43:11 +03:00
template < typename T , typename Arg1 , typename Arg2 , typename Arg3 , typename Arg4 >
std : : unique_ptr < T > make_unique ( Arg1 & & arg1 , Arg2 & & arg2 , Arg3 & & arg3 , Arg4 & & arg4 )
{
return std : : unique_ptr < T > ( new T ( std : : forward < Arg1 > ( arg1 ) , std : : forward < Arg2 > ( arg2 ) , std : : forward < Arg3 > ( arg3 ) , std : : forward < Arg4 > ( arg4 ) ) ) ;
}
2013-07-02 15:08:30 +03:00
# endif
2012-05-05 00:16:39 +03:00
template < typename Container >
typename Container : : const_reference circularAt ( const Container & r , size_t index )
{
assert ( r . size ( ) ) ;
index % = r . size ( ) ;
2013-06-29 16:05:48 +03:00
auto itr = std : : begin ( r ) ;
2012-05-05 00:16:39 +03:00
std : : advance ( itr , index ) ;
return * itr ;
}
2012-08-26 12:59:07 +03:00
2012-08-26 12:07:48 +03:00
template < typename Range , typename Predicate >
void erase_if ( Range & vec , Predicate pred )
{
vec . erase ( boost : : remove_if ( vec , pred ) , vec . end ( ) ) ;
}
2012-09-24 02:10:56 +03:00
template < typename Elem , typename Predicate >
void erase_if ( std : : set < Elem > & setContainer , Predicate pred )
{
auto itr = setContainer . begin ( ) ;
auto endItr = setContainer . end ( ) ;
while ( itr ! = endItr )
{
auto tmpItr = itr + + ;
if ( pred ( * tmpItr ) )
setContainer . erase ( tmpItr ) ;
2013-12-03 18:24:13 +03:00
}
}
//works for map and std::map, maybe something else
template < typename Key , typename Val , typename Predicate >
void erase_if ( std : : map < Key , Val > & container , Predicate pred )
{
2012-09-29 20:36:48 +03:00
auto itr = container . begin ( ) ;
auto endItr = container . end ( ) ;
while ( itr ! = endItr )
{
auto tmpItr = itr + + ;
if ( pred ( * tmpItr ) )
container . erase ( tmpItr ) ;
}
}
2012-08-26 12:07:48 +03:00
template < typename InputRange , typename OutputIterator , typename Predicate >
OutputIterator copy_if ( const InputRange & input , OutputIterator result , Predicate pred )
{
2013-06-29 16:05:48 +03:00
return std : : copy_if ( boost : : const_begin ( input ) , std : : end ( input ) , result , pred ) ;
2012-08-26 12:07:48 +03:00
}
template < typename Container >
std : : insert_iterator < Container > set_inserter ( Container & c )
{
return std : : inserter ( c , c . end ( ) ) ;
}
2012-09-20 19:55:21 +03:00
//Returns iterator to the element for which the value of ValueFunction is minimal
2012-08-26 12:07:48 +03:00
template < class ForwardRange , class ValueFunction >
2013-06-29 16:05:48 +03:00
auto minElementByFun ( const ForwardRange & rng , ValueFunction vf ) - > decltype ( std : : begin ( rng ) )
2012-08-26 12:07:48 +03:00
{
2014-02-05 23:25:36 +03:00
/* Clang crashes when instantiating this function template and having PCH compilation enabled.
* There is a bug report here : http : //llvm.org/bugs/show_bug.cgi?id=18744
* Current bugfix is to don ' t use a typedef for decltype ( * std : : begin ( rng ) ) and to use decltype
* directly for both function parameters .
*/
return boost : : min_element ( rng , [ & ] ( decltype ( * std : : begin ( rng ) ) lhs , decltype ( * std : : begin ( rng ) ) rhs ) - > bool
2012-08-26 12:07:48 +03:00
{
return vf ( lhs ) < vf ( rhs ) ;
} ) ;
}
2012-09-20 19:55:21 +03:00
//Returns iterator to the element for which the value of ValueFunction is maximal
template < class ForwardRange , class ValueFunction >
2013-06-29 16:05:48 +03:00
auto maxElementByFun ( const ForwardRange & rng , ValueFunction vf ) - > decltype ( std : : begin ( rng ) )
2012-09-20 19:55:21 +03:00
{
2014-02-05 23:25:36 +03:00
/* Clang crashes when instantiating this function template and having PCH compilation enabled.
* There is a bug report here : http : //llvm.org/bugs/show_bug.cgi?id=18744
* Current bugfix is to don ' t use a typedef for decltype ( * std : : begin ( rng ) ) and to use decltype
* directly for both function parameters .
*/
return boost : : max_element ( rng , [ & ] ( decltype ( * std : : begin ( rng ) ) lhs , decltype ( * std : : begin ( rng ) ) rhs ) - > bool
2012-09-20 19:55:21 +03:00
{
return vf ( lhs ) < vf ( rhs ) ;
} ) ;
}
2013-02-04 15:32:53 +03:00
2013-02-07 02:24:43 +03:00
template < typename T >
void advance ( T & obj , int change )
{
obj = ( T ) ( ( ( int ) obj ) + change ) ;
}
2013-04-20 14:34:01 +03:00
template < typename Container >
2013-06-26 14:18:27 +03:00
typename Container : : value_type backOrNull ( const Container & c ) //returns last element of container or nullptr if it is empty (to be used with containers of pointers)
2013-04-20 14:34:01 +03:00
{
if ( c . size ( ) )
return c . back ( ) ;
else
2013-06-20 00:26:27 +03:00
return typename Container : : value_type ( ) ;
2013-04-20 14:34:01 +03:00
}
template < typename Container >
2013-06-26 14:18:27 +03:00
typename Container : : value_type frontOrNull ( const Container & c ) //returns first element of container or nullptr if it is empty (to be used with containers of pointers)
2013-04-20 14:34:01 +03:00
{
if ( c . size ( ) )
return c . front ( ) ;
else
2013-04-21 15:49:26 +03:00
return nullptr ;
2013-04-20 14:34:01 +03:00
}
2013-07-07 11:27:27 +03:00
2013-09-06 01:11:13 +03:00
template < typename Container , typename Index >
bool isValidIndex ( const Container & c , Index i )
{
return i > = 0 & & i < c . size ( ) ;
}
template < typename Container , typename Index >
boost : : optional < typename Container : : const_reference > tryAt ( const Container & c , Index i )
{
if ( isValidIndex ( c , i ) )
{
auto itr = c . begin ( ) ;
std : : advance ( itr , i ) ;
return * itr ;
}
return boost : : none ;
}
template < typename Container , typename Pred >
static boost : : optional < typename Container : : const_reference > tryFindIf ( const Container & r , const Pred & t )
{
auto pos = range : : find_if ( r , t ) ;
if ( pos = = boost : : end ( r ) )
return boost : : none ;
else
return * pos ;
}
2013-09-12 00:57:08 +03:00
template < typename Container >
typename Container : : const_reference atOrDefault ( const Container & r , size_t index , const typename Container : : const_reference & defaultValue )
{
2013-09-21 21:29:26 +03:00
if ( index < r . size ( ) )
2013-09-12 00:57:08 +03:00
return r [ index ] ;
return defaultValue ;
}
2013-07-07 11:27:27 +03:00
using boost : : math : : round ;
2012-03-13 15:47:47 +03:00
}
using vstd : : operator - = ;
2013-04-05 13:29:46 +03:00
using vstd : : make_unique ;