diff --git a/Telegram/SourceFiles/rpl/combine_previous.h b/Telegram/SourceFiles/rpl/combine_previous.h new file mode 100644 index 000000000..6c4a6c826 --- /dev/null +++ b/Telegram/SourceFiles/rpl/combine_previous.h @@ -0,0 +1,126 @@ +/* +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 + +#include +#include "base/optional.h" + +namespace rpl { +namespace details { + +class combine_previous_helper { +public: + template + rpl::producer, Error> operator()( + rpl::producer &&initial) const { + using consumer_type = consumer< + std::tuple, + Error>; + return [initial = std::move(initial)]( + const consumer_type &consumer) mutable { + auto previous = consumer.template make_state< + base::optional + >(); + return std::move(initial).start( + [consumer, previous](auto &&value) { + if (auto exists = *previous) { + &existing = *exists; + auto next = std::make_tuple( + std::move(existing), + value); + consumer.put_next(std::move(next)); + existing = std::forward( + value); + } else { + *previous = std::forward( + value); + } + }, [consumer](auto &&error) { + consumer.put_error_forward(std::forward(error)); + }, [consumer] { + consumer.put_done(); + }); + }; + } + +}; + +template +class combine_previous_with_default_helper { +public: + template + combine_previous_with_default_helper(OtherValue &&value) + : _value(std::forward(value)) { + } + + template + rpl::producer, Error> operator()( + rpl::producer &&initial) { + using consumer_type = consumer< + std::tuple, + Error>; + return [ + initial = std::move(initial), + value = Value(std::move(_value)) + ](const consumer_type &consumer) mutable { + auto previous = consumer.template make_state( + std::move(value)); + return std::move(initial).start( + [consumer, previous](auto &&value) { + auto &existing = *previous; + auto next = std::make_tuple( + std::move(existing), + value); + consumer.put_next(std::move(next)); + existing = std::forward(value); + }, [consumer](auto &&error) { + consumer.put_error_forward(std::forward(error)); + }, [consumer] { + consumer.put_done(); + }); + }; + } + +private: + DefaultValue _value; + +}; + +template +combine_previous_with_default_helper> +combine_previous_with_default(DefaultValue &&value) { + return { std::forward(value) }; +} + +} // namespace details + +inline auto combine_previous() +-> details::combine_previous_helper { + return details::combine_previous_helper(); +} + +template +inline auto combine_previous(DefaultValue &&value) { + return details::combine_previous_with_default( + std::forward(value)); +} + +} // namespace rpl diff --git a/Telegram/gyp/tests/tests.gyp b/Telegram/gyp/tests/tests.gyp index ef60158e0..e2524264e 100644 --- a/Telegram/gyp/tests/tests.gyp +++ b/Telegram/gyp/tests/tests.gyp @@ -102,6 +102,7 @@ '<(src_loc)/rpl/after_next.h', '<(src_loc)/rpl/before_next.h', '<(src_loc)/rpl/combine.h', + '<(src_loc)/rpl/combine_previous.h', '<(src_loc)/rpl/complete.h', '<(src_loc)/rpl/consumer.h', '<(src_loc)/rpl/deferred.h',