2013-06-29 13:05:48 +00:00
# pragma once
2012-03-13 12:47:47 +00: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 10:29:46 +00: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. " ) ;
# ifdef __GNUC__
2013-06-29 13:05:48 +00: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 10:29:46 +00: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 10:29:46 +00:00
# endif
/* ---------------------------------------------------------------------------- */
/* Suppress some compiler warnings */
/* ---------------------------------------------------------------------------- */
2013-02-03 21:05:44 +00:00
# ifdef _MSC_VER
2013-06-29 13:05:48 +00:00
# pragma warning (disable : 4800 ) /* disable conversion to bool warning -- I think it's intended in all places */
2013-04-05 10:29:46 +00:00
# endif
2013-02-03 21:05:44 +00:00
2014-08-10 23:42:39 +02:00
/* ---------------------------------------------------------------------------- */
/* System detection. */
/* ---------------------------------------------------------------------------- */
// Based on: http://sourceforge.net/p/predef/wiki/OperatingSystems/
// and on: http://stackoverflow.com/questions/5919996/how-to-detect-reliably-mac-os-x-ios-linux-windows-in-c-preprocessor
// TODO?: Should be moved to vstd\os_detect.h (and then included by Global.h)
# ifdef _WIN16 // Defined for 16-bit environments
2014-08-30 15:04:00 +02:00
# error "16-bit Windows isn't supported"
2014-08-10 23:42:39 +02:00
# elif defined(_WIN64) // Defined for 64-bit environments
2014-08-30 15:04:00 +02:00
# define VCMI_WINDOWS
# define VCMI_WINDOWS_64
2014-08-10 23:42:39 +02:00
# elif defined(_WIN32) // Defined for both 32-bit and 64-bit environments
2014-08-30 15:04:00 +02:00
# define VCMI_WINDOWS
# define VCMI_WINDOWS_32
2014-08-10 23:42:39 +02:00
# elif defined(_WIN32_WCE)
2014-08-30 15:04:00 +02:00
# error "Windows CE isn't supported"
2014-08-10 23:42:39 +02:00
# elif defined(__linux__) || defined(__gnu_linux__) || defined(linux) || defined(__linux)
2014-08-30 15:04:00 +02:00
# define VCMI_UNIX
2014-11-17 12:19:29 +01:00
# define VCMI_XDG
2014-08-30 15:04:00 +02:00
# ifdef __ANDROID__
2016-02-15 13:34:37 +03:00
# define VCMI_ANDROID
2014-08-30 15:04:00 +02:00
# endif
2014-11-17 12:25:10 +01:00
# elif defined(__FreeBSD_kernel__) || defined(__FreeBSD__)
# define VCMI_UNIX
# define VCMI_XDG
# define VCMI_FREEBSD
2015-10-12 08:33:52 +03:00
# elif defined(__GNU__) || defined(__gnu_hurd__) || (defined(__MACH__) && !defined(__APPLE__))
2014-11-17 13:15:50 +01:00
# define VCMI_UNIX
# define VCMI_XDG
# define VCMI_HURD
2014-08-10 23:42:39 +02:00
# elif defined(__APPLE__) && defined(__MACH__)
2014-08-30 15:04:00 +02:00
# define VCMI_UNIX
# define VCMI_APPLE
# include "TargetConditionals.h"
# if TARGET_IPHONE_SIMULATOR
# define VCMI_IOS
# define VCMI_IOS_SIM
# elif TARGET_OS_IPHONE
# define VCMI_IOS
# elif TARGET_OS_MAC
# define VCMI_MAC
# else
//# warning "Unknown Apple target."?
# endif
2014-08-10 23:42:39 +02:00
# else
2014-08-30 15:04:00 +02:00
# error "VCMI supports only Windows, OSX, Linux and Android targets"
2014-08-10 23:42:39 +02:00
# endif
# ifdef VCMI_IOS
2014-08-30 15:04:00 +02:00
# error "iOS system isn't yet supported."
2014-08-10 23:42:39 +02:00
# endif
2013-04-05 10:29:46 +00:00
/* ---------------------------------------------------------------------------- */
/* Commonly used C++, Boost headers */
/* ---------------------------------------------------------------------------- */
2014-08-27 12:31:58 +02:00
# ifdef VCMI_WINDOWS
2014-08-30 15:04:00 +02:00
# define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers - delete this line if something is missing.
# define NOMINMAX // Exclude min/max macros from <Windows.h>. Use std::[min/max] from <algorithm> instead.
2015-12-22 01:57:56 +03:00
# define _NO_W32_PSEUDO_MODIFIERS // Exclude more macros for compiling with MinGW on Linux.
2014-08-21 22:26:28 +02:00
# endif
2015-11-22 07:14:52 +03:00
/* ---------------------------------------------------------------------------- */
/* A macro to force inlining some of our functions */
/* ---------------------------------------------------------------------------- */
// Compiler (at least MSVC) is not so smart here-> without that displaying is MUCH slower
# ifdef _MSC_VER
# define STRONG_INLINE __forceinline
# elif __GNUC__
# define STRONG_INLINE inline __attribute__((always_inline))
# else
# define STRONG_INLINE inline
# endif
2013-08-02 08:28:52 +00:00
# define _USE_MATH_DEFINES
2012-03-13 12:47:47 +00:00
# include <cstdio>
# include <stdio.h>
2012-08-08 08:25:27 +00:00
# include <algorithm>
# include <array>
2012-03-13 12:47:47 +00:00
# include <cassert>
2012-08-08 08:25:27 +00:00
# include <climits>
# include <cmath>
# include <cstdlib>
2013-06-26 11:18:27 +00:00
# include <functional>
2012-08-08 08:25:27 +00:00
# include <fstream>
# include <iomanip>
# include <iostream>
2012-03-13 12:47:47 +00:00
# include <map>
2012-08-08 08:25:27 +00:00
# include <memory>
# include <numeric>
2012-03-13 12:47:47 +00:00
# include <queue>
2013-06-26 11:18:27 +00:00
# include <random>
2012-03-13 12:47:47 +00:00
# include <set>
2012-08-08 08:25:27 +00:00
# include <sstream>
2012-08-26 09:07:48 +00:00
# include <string>
2013-06-29 13:05:48 +00:00
# include <unordered_set>
# include <unordered_map>
2012-03-13 12:47:47 +00:00
# include <utility>
2012-08-26 09:07:48 +00:00
# include <vector>
2012-03-13 12:47:47 +00:00
2012-07-02 09:04:05 +00:00
//The only available version is 3, as of Boost 1.50
2012-09-16 15:49:22 +00:00
# include <boost/version.hpp>
2012-07-02 09:04:05 +00:00
# define BOOST_FILESYSTEM_VERSION 3
2014-02-05 20:25:36 +00:00
# if BOOST_VERSION > 105000
2014-08-30 15:04:00 +02:00
# define BOOST_THREAD_VERSION 3
2012-09-16 15:49:22 +00:00
# endif
2012-09-15 19:16:16 +00:00
# define BOOST_THREAD_DONT_PROVIDE_THREAD_DESTRUCTOR_CALLS_TERMINATE_IF_JOINABLE 1
2014-08-04 20:33:59 +02:00
# define BOOST_BIND_NO_PLACEHOLDERS
2012-03-13 12:47:47 +00:00
# include <boost/algorithm/string.hpp>
# include <boost/cstdint.hpp>
2013-04-10 16:28:14 +00:00
# include <boost/current_function.hpp>
2013-11-08 20:36:26 +00:00
# include <boost/crc.hpp>
2012-03-13 12:47:47 +00:00
# include <boost/date_time/posix_time/posix_time.hpp>
2013-04-05 10:29:46 +00:00
# include <boost/date_time/posix_time/posix_time_io.hpp>
2012-03-13 12:47:47 +00:00
# include <boost/filesystem.hpp>
2014-08-10 23:42:39 +02:00
# include <boost/filesystem/path.hpp>
# include <boost/filesystem/fstream.hpp>
2012-03-13 12:47:47 +00:00
# include <boost/format.hpp>
2014-07-07 11:02:55 +03:00
# include <boost/functional/hash.hpp>
2012-03-13 12:47:47 +00:00
# include <boost/lexical_cast.hpp>
2014-11-02 16:19:14 +03:00
# ifndef VCMI_ANDROID
2014-08-21 22:26:28 +02:00
# include <boost/locale/generator.hpp>
2014-11-02 16:19:14 +03:00
# endif
2012-03-13 12:47:47 +00:00
# include <boost/logic/tribool.hpp>
2012-08-26 09:07:48 +00:00
# include <boost/optional.hpp>
2015-10-06 08:56:22 +03:00
# include <boost/optional/optional_io.hpp>
2013-06-26 11:18:27 +00:00
# include <boost/program_options.hpp>
2013-02-09 19:19:14 +00:00
# include <boost/range/adaptor/filtered.hpp>
2013-06-29 13:05:48 +00:00
# include <boost/range/adaptor/reversed.hpp>
2013-06-26 11:18:27 +00:00
# include <boost/range/algorithm.hpp>
2012-03-13 12:47:47 +00:00
# include <boost/thread.hpp>
2012-04-14 02:20:22 +00:00
# include <boost/variant.hpp>
2013-07-07 08:27:27 +00:00
# include <boost/math/special_functions/round.hpp>
2015-08-31 07:39:03 +02:00
# include <boost/multi_array.hpp>
2013-07-07 08:27:27 +00:00
2014-07-09 17:42:22 +03:00
# ifndef M_PI
2014-08-27 12:31:58 +02:00
# define M_PI 3.14159265358979323846
2014-07-09 17:42:22 +03:00
# endif
2013-04-05 10:29:46 +00:00
/* ---------------------------------------------------------------------------- */
/* Usings */
/* ---------------------------------------------------------------------------- */
2014-08-04 20:33:59 +02:00
using namespace std : : placeholders ;
2013-04-05 10:29:46 +00:00
namespace range = boost : : range ;
/* ---------------------------------------------------------------------------- */
/* Typedefs */
/* ---------------------------------------------------------------------------- */
2012-03-13 12:47:47 +00: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 10:29:46 +00:00
// Lock typedefs
typedef boost : : lock_guard < boost : : mutex > TLockGuard ;
2013-04-09 14:31:36 +00:00
typedef boost : : lock_guard < boost : : recursive_mutex > TLockGuardRec ;
2012-03-13 12:47:47 +00:00
2013-04-05 10:29:46 +00:00
/* ---------------------------------------------------------------------------- */
/* Macros */
/* ---------------------------------------------------------------------------- */
2012-03-13 12:47:47 +00:00
// Import + Export macro declarations
2014-08-21 22:26:28 +02:00
# ifdef VCMI_WINDOWS
2013-06-26 11:18:27 +00:00
# ifdef __GNUC__
2014-08-30 15:04:00 +02:00
# define DLL_IMPORT __attribute__((dllimport))
2013-06-26 11:18:27 +00:00
# define DLL_EXPORT __attribute__((dllexport))
# else
2014-08-30 15:04:00 +02:00
# define DLL_IMPORT __declspec(dllimport)
2013-06-26 11:18:27 +00:00
# define DLL_EXPORT __declspec(dllexport)
# endif
2014-06-26 18:34:54 +03:00
# define ELF_VISIBILITY
2012-03-13 12:47:47 +00:00
# else
2013-06-26 11:18:27 +00:00
# ifdef __GNUC__
2014-08-30 15:04:00 +02:00
# define DLL_IMPORT __attribute__ ((visibility("default")))
2014-06-26 18:34:54 +03:00
# define DLL_EXPORT __attribute__ ((visibility("default")))
# define ELF_VISIBILITY __attribute__ ((visibility("default")))
# define ELF_VISIBILITY __attribute__ ((visibility("default")))
2013-06-26 11:18:27 +00:00
# endif
2012-03-13 12:47:47 +00:00
# endif
# ifdef VCMI_DLL
2013-06-26 11:18:27 +00:00
# define DLL_LINKAGE DLL_EXPORT
2012-03-13 12:47:47 +00:00
# else
2013-06-26 11:18:27 +00:00
# define DLL_LINKAGE DLL_IMPORT
2012-03-13 12:47:47 +00:00
# endif
2013-04-05 10:29:46 +00:00
# define THROW_FORMAT(message, formatting_elems) throw std::runtime_error(boost::str(boost::format(message) % formatting_elems))
2012-03-13 12:47:47 +00:00
2013-04-11 15:58:01 +00:00
# define ASSERT_IF_CALLED_WITH_PLAYER if(!player) {logGlobal->errorStream() << BOOST_CURRENT_FUNCTION; assert(0);}
2012-03-13 12:47:47 +00:00
2013-04-05 10:29:46 +00: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 12:47:47 +00:00
2013-11-09 13:49:36 +00: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 15:24:13 +00:00
/* ---------------------------------------------------------------------------- */
/* VCMI standard library */
/* ---------------------------------------------------------------------------- */
2013-12-28 12:47:55 +00: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 15:24:13 +00:00
template < typename T >
2013-12-08 17:54:13 +00:00
std : : ostream & operator < < ( std : : ostream & out , const boost : : optional < T > & opt )
2013-12-03 15:24:13 +00:00
{
2013-12-08 17:54:13 +00:00
if ( opt ) return out < < * opt ;
else return out < < " empty " ;
2013-08-18 15:46:28 +00:00
}
2013-12-08 17:54:13 +00: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 15:46:28 +00:00
2012-03-13 12:47:47 +00:00
namespace vstd
{
2016-02-15 13:34:37 +03:00
2013-06-29 13:05:48 +00: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 ) ;
}
2016-02-15 13:34:37 +03:00
2012-03-13 12:47:47 +00: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 13:05:48 +00:00
return std : : find ( std : : begin ( c ) , std : : end ( c ) , i ) ! = std : : end ( c ) ;
2012-08-26 09:07:48 +00: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 13:05:48 +00:00
return std : : find_if ( std : : begin ( c ) , std : : end ( c ) , p ) ! = std : : end ( c ) ;
2012-03-13 12:47:47 +00: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 15:24:13 +00: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 12:47:47 +00:00
{
return c . find ( i ) ! = c . end ( ) ;
}
2013-03-03 18:00:37 +00:00
template < typename V , typename Item , typename Item2 >
2013-06-29 13:05:48 +00:00
bool contains ( const std : : unordered_map < Item , V > & c , const Item2 & i )
2013-03-03 18:00:37 +00:00
{
return c . find ( i ) ! = c . end ( ) ;
}
2012-03-13 12:47:47 +00:00
//returns position of first element in vector c equal to s, if there is no such element, -1 is returned
2012-06-13 13:04:06 +00:00
template < typename Container , typename T2 >
int find_pos ( const Container & c , const T2 & s )
2012-03-13 12:47:47 +00:00
{
2012-06-13 13:04:06 +00:00
size_t i = 0 ;
2013-06-29 13:05:48 +00:00
for ( auto iter = std : : begin ( c ) ; iter ! = std : : end ( c ) ; iter + + , i + + )
2012-06-13 13:04:06 +00:00
if ( * iter = = s )
2012-03-13 12:47:47 +00:00
return i ;
return - 1 ;
}
2013-07-21 22:01:29 +00: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 12:47:47 +00:00
{
2013-07-21 22:01:29 +00:00
auto ret = boost : : range : : find_if ( c , f ) ;
if ( ret ! = std : : end ( c ) )
return std : : distance ( std : : begin ( c ) , ret ) ;
2012-03-13 12:47:47 +00: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 07:34:37 +00:00
bool isbetween ( const t1 & value , const t2 & min , const t3 & max )
2012-03-13 12:47:47 +00:00
{
2012-08-06 07:34:37 +00:00
return value > min & & value < max ;
2012-03-13 12:47:47 +00:00
}
//checks if a is within b and c
template < typename t1 , typename t2 , typename t3 >
2012-08-06 07:34:37 +00:00
bool iswithin ( const t1 & value , const t2 & min , const t3 & max )
2012-03-13 12:47:47 +00:00
{
2012-08-06 07:34:37 +00:00
return value > = min & & value < = max ;
2012-03-13 12:47:47 +00: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 11:18:27 +00:00
//deleted pointer and sets it to nullptr
2012-03-13 12:47:47 +00:00
template < typename T >
void clear_pointer ( T * & ptr )
{
delete ptr ;
2013-06-26 11:18:27 +00:00
ptr = nullptr ;
2012-03-13 12:47:47 +00:00
}
2013-07-02 12:08:30 +00:00
# if _MSC_VER >= 1800
using std : : make_unique ;
# else
2012-02-19 21:03:43 +00:00
template < typename T >
std : : unique_ptr < T > make_unique ( )
{
return std : : unique_ptr < T > ( new T ( ) ) ;
2012-03-13 12:47:47 +00:00
}
2012-02-19 21:03:43 +00:00
template < typename T , typename Arg1 >
2012-04-08 01:15:18 +00:00
std : : unique_ptr < T > make_unique ( Arg1 & & arg1 )
2012-02-19 21:03:43 +00:00
{
return std : : unique_ptr < T > ( new T ( std : : forward < Arg1 > ( arg1 ) ) ) ;
2012-03-13 12:47:47 +00:00
}
2012-04-08 01:15:18 +00: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 14:43:58 +00: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 11:43:11 +00: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 12:08:30 +00:00
# endif
2012-05-04 21:16:39 +00:00
template < typename Container >
typename Container : : const_reference circularAt ( const Container & r , size_t index )
{
assert ( r . size ( ) ) ;
index % = r . size ( ) ;
2013-06-29 13:05:48 +00:00
auto itr = std : : begin ( r ) ;
2012-05-04 21:16:39 +00:00
std : : advance ( itr , index ) ;
return * itr ;
}
2012-08-26 09:59:07 +00:00
2012-08-26 09:07:48 +00:00
template < typename Range , typename Predicate >
void erase_if ( Range & vec , Predicate pred )
{
vec . erase ( boost : : remove_if ( vec , pred ) , vec . end ( ) ) ;
}
2012-09-23 23:10:56 +00:00
template < typename Elem , typename Predicate >
void erase_if ( std : : set < Elem > & setContainer , Predicate pred )
{
auto itr = setContainer . begin ( ) ;
2016-02-15 13:34:37 +03:00
auto endItr = setContainer . end ( ) ;
2012-09-23 23:10:56 +00:00
while ( itr ! = endItr )
{
auto tmpItr = itr + + ;
if ( pred ( * tmpItr ) )
setContainer . erase ( tmpItr ) ;
2013-12-03 15:24:13 +00: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 17:36:48 +00:00
auto itr = container . begin ( ) ;
2016-02-15 13:34:37 +03:00
auto endItr = container . end ( ) ;
2012-09-29 17:36:48 +00:00
while ( itr ! = endItr )
{
auto tmpItr = itr + + ;
if ( pred ( * tmpItr ) )
container . erase ( tmpItr ) ;
}
}
2012-08-26 09:07:48 +00:00
template < typename InputRange , typename OutputIterator , typename Predicate >
OutputIterator copy_if ( const InputRange & input , OutputIterator result , Predicate pred )
{
2013-06-29 13:05:48 +00:00
return std : : copy_if ( boost : : const_begin ( input ) , std : : end ( input ) , result , pred ) ;
2012-08-26 09:07:48 +00:00
}
template < typename Container >
std : : insert_iterator < Container > set_inserter ( Container & c )
{
return std : : inserter ( c , c . end ( ) ) ;
}
2012-09-20 16:55:21 +00:00
//Returns iterator to the element for which the value of ValueFunction is minimal
2012-08-26 09:07:48 +00:00
template < class ForwardRange , class ValueFunction >
2013-06-29 13:05:48 +00:00
auto minElementByFun ( const ForwardRange & rng , ValueFunction vf ) - > decltype ( std : : begin ( rng ) )
2012-08-26 09:07:48 +00:00
{
2014-02-05 20:25:36 +00: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 09:07:48 +00:00
{
return vf ( lhs ) < vf ( rhs ) ;
} ) ;
}
2016-02-15 13:34:37 +03:00
2012-09-20 16:55:21 +00:00
//Returns iterator to the element for which the value of ValueFunction is maximal
template < class ForwardRange , class ValueFunction >
2013-06-29 13:05:48 +00:00
auto maxElementByFun ( const ForwardRange & rng , ValueFunction vf ) - > decltype ( std : : begin ( rng ) )
2012-09-20 16:55:21 +00:00
{
2014-02-05 20:25:36 +00: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 16:55:21 +00:00
{
return vf ( lhs ) < vf ( rhs ) ;
} ) ;
}
2013-02-04 12:32:53 +00:00
2013-02-06 23:24:43 +00:00
template < typename T >
void advance ( T & obj , int change )
{
obj = ( T ) ( ( ( int ) obj ) + change ) ;
}
2013-04-20 11:34:01 +00:00
template < typename Container >
2013-06-26 11:18:27 +00: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 11:34:01 +00:00
{
if ( c . size ( ) )
return c . back ( ) ;
else
2013-06-19 21:26:27 +00:00
return typename Container : : value_type ( ) ;
2013-04-20 11:34:01 +00:00
}
template < typename Container >
2013-06-26 11:18:27 +00: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 11:34:01 +00:00
{
if ( c . size ( ) )
return c . front ( ) ;
else
2013-04-21 12:49:26 +00:00
return nullptr ;
2013-04-20 11:34:01 +00:00
}
2013-07-07 08:27:27 +00:00
2013-09-05 22:11:13 +00: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-11 21:57:08 +00:00
template < typename Container >
typename Container : : const_reference atOrDefault ( const Container & r , size_t index , const typename Container : : const_reference & defaultValue )
{
2013-09-21 18:29:26 +00:00
if ( index < r . size ( ) )
2013-09-11 21:57:08 +00:00
return r [ index ] ;
2016-02-15 13:34:37 +03:00
2013-09-11 21:57:08 +00:00
return defaultValue ;
}
2014-06-05 17:19:11 +02:00
template < typename Container , typename Item >
bool erase_if_present ( Container & c , const Item & item )
{
auto i = std : : find ( c . begin ( ) , c . end ( ) , item ) ;
if ( i ! = c . end ( ) )
{
c . erase ( i ) ;
return true ;
}
return false ;
}
template < typename V , typename Item , typename Item2 >
bool erase_if_present ( std : : map < Item , V > & c , const Item2 & item )
{
auto i = c . find ( item ) ;
if ( i ! = c . end ( ) )
{
c . erase ( i ) ;
return true ;
}
return false ;
}
template < typename Container , typename Pred >
void erase ( Container & c , Pred pred )
{
c . erase ( boost : : remove_if ( c , pred ) , c . end ( ) ) ;
}
template < typename T >
void removeDuplicates ( std : : vector < T > & vec )
{
boost : : sort ( vec ) ;
vec . erase ( std : : unique ( vec . begin ( ) , vec . end ( ) ) , vec . end ( ) ) ;
}
2016-02-15 13:34:37 +03:00
2014-10-04 00:34:13 +04:00
template < typename T >
void concatenate ( std : : vector < T > & dest , const std : : vector < T > & src )
{
dest . reserve ( dest . size ( ) + src . size ( ) ) ;
2016-02-15 13:34:37 +03:00
dest . insert ( dest . end ( ) , src . begin ( ) , src . end ( ) ) ;
2014-10-04 00:34:13 +04:00
}
2014-06-05 17:19:11 +02:00
2015-03-08 14:41:17 +03:00
template < typename T >
std : : vector < T > intersection ( std : : vector < T > & v1 , std : : vector < T > & v2 )
{
std : : vector < T > v3 ;
std : : sort ( v1 . begin ( ) , v1 . end ( ) ) ;
std : : sort ( v2 . begin ( ) , v2 . end ( ) ) ;
2015-03-08 20:19:00 +03:00
std : : set_intersection ( v1 . begin ( ) , v1 . end ( ) , v2 . begin ( ) , v2 . end ( ) , std : : back_inserter ( v3 ) ) ;
2015-03-08 14:41:17 +03:00
return v3 ;
}
2013-07-07 08:27:27 +00:00
using boost : : math : : round ;
2012-03-13 12:47:47 +00:00
}
using vstd : : operator - = ;
2013-04-05 10:29:46 +00:00
using vstd : : make_unique ;