/* This file is part of Telegram Desktop, the official desktop version of Telegram messaging app, see https://telegram.org Telegram Desktop is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. It 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 General Public License for more details. In addition, as a special exception, the copyright holders give permission to link the code of portions of this program with the OpenSSL library. Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #pragma once namespace base { template inline Type take(Type &value) { return std::exchange(value, Type {}); } template inline constexpr size_t array_size(const Type(&)[Size]) { return Size; } template decltype(auto) for_each(Range &&range, Method &&method) { return std::for_each( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(method)); } template decltype(auto) for_each_apply(Method &&method) { return [&method](auto &&range) { return for_each(std::forward(range), std::forward(method)); }; } template decltype(auto) find(Range &&range, Value &&value) { return std::find( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(value)); } template decltype(auto) find_if(Range &&range, Predicate &&predicate) { return std::find_if( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(predicate)); } template decltype(auto) lower_bound(Range &&range, Type &&value) { return std::lower_bound( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(value)); } template decltype(auto) lower_bound(Range &&range, Type &&value, Predicate &&predicate) { return std::lower_bound( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(value), std::forward(predicate)); } template decltype(auto) upper_bound(Range &&range, Type &&value) { return std::upper_bound( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(value)); } template decltype(auto) upper_bound(Range &&range, Type &&value, Predicate &&predicate) { return std::upper_bound( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(value), std::forward(predicate)); } template decltype(auto) equal_range(Range &&range, Type &&value) { return std::equal_range( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(value)); } template decltype(auto) equal_range(Range &&range, Type &&value, Predicate &&predicate) { return std::equal_range( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(value), std::forward(predicate)); } template decltype(auto) sort(Range &&range) { return std::sort( std::begin(std::forward(range)), std::end(std::forward(range))); } template decltype(auto) sort(Range &&range, Predicate &&predicate) { return std::sort( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(predicate)); } template decltype(auto) stable_partition(Range &&range, Predicate &&predicate) { return std::stable_partition( std::begin(std::forward(range)), std::end(std::forward(range)), std::forward(predicate)); } } // namespace base