@file:Suppress("unused") package dev.inmo.tgbotapi.extensions.behaviour_builder.expectations import dev.inmo.micro_utils.coroutines.safelyWithoutExceptions import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContext import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.SimpleFilter import dev.inmo.tgbotapi.extensions.utils.withContent import dev.inmo.tgbotapi.requests.abstracts.Request import dev.inmo.tgbotapi.types.message.abstracts.CommonMessage import dev.inmo.tgbotapi.types.message.content.* import dev.inmo.tgbotapi.types.message.content.AudioMediaGroupContent import dev.inmo.tgbotapi.types.message.content.DocumentMediaGroupContent import dev.inmo.tgbotapi.types.message.content.MediaGroupContent import dev.inmo.tgbotapi.types.message.content.VisualMediaGroupContent import dev.inmo.tgbotapi.types.message.content.InvoiceContent import dev.inmo.tgbotapi.types.update.MediaGroupUpdates.SentMediaGroupUpdate import dev.inmo.tgbotapi.types.update.abstracts.BaseSentMessageUpdate import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.toList typealias CommonMessageToCommonMessageMapper = suspend CommonMessage.() -> CommonMessage? internal suspend fun BehaviourContext.waitCommonMessage( count: Int = 1, initRequest: Request<*>? = null, includeMediaGroups: Boolean = true, errorFactory: NullableRequestBuilder<*> = { null }, filter: SimpleFilter>? = null, mapper: suspend CommonMessage.() -> CommonMessage? ): Flow> = expectFlow( initRequest, count, errorFactory ) { val messages = when (it) { is SentMediaGroupUpdate -> { if (includeMediaGroups) { it.data.map { it as CommonMessage } } else { emptyList() } } is BaseSentMessageUpdate -> listOf(it.data) else -> return@expectFlow emptyList() } messages.mapNotNull { message -> val asCommonMessage = message as CommonMessage if (filter == null || filter(asCommonMessage)) { asCommonMessage.mapper() } else { null } } } internal inline fun contentMessageConverter( noinline mapper: CommonMessageToCommonMessageMapper? = null ): suspend CommonMessage.() -> CommonMessage? = mapper ?.let { { if (content is T) { @Suppress("UNCHECKED_CAST") val message = (this as CommonMessage) safelyWithoutExceptions { mapper(message) } } else { null } } } ?: { @Suppress("UNCHECKED_CAST") if (content is T) this as CommonMessage else null } private suspend inline fun BehaviourContext.waitContentMessage( count: Int = 1, initRequest: Request<*>? = null, includeMediaGroups: Boolean = true, noinline errorFactory: NullableRequestBuilder<*> = { null }, filter: SimpleFilter>? = null, noinline mapper: CommonMessageToCommonMessageMapper? = null ) : List> = waitCommonMessage( count, initRequest, includeMediaGroups, errorFactory, filter ?.let { { it.withContent() ?.let { filter(it) } == true } }, contentMessageConverter(mapper) ).toList() suspend fun BehaviourContext.waitContentMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = true, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitContactMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitDiceMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitGameMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitLocationMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitLiveLocationMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitStaticLocationMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitPollMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitTextMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitVenueMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitAudioMediaGroupContentMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = true, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitDocumentMediaGroupContentMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = true, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitMediaMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = false, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitAnyMediaGroupContentMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = true, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitVisualMediaGroupContentMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = true, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitTextedMediaContentMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = true, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitAnimationMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitAudioMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = false, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitDocumentMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = false, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitPhotoMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = false, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitStickerMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitVideoMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, includeMediaGroups: Boolean = false, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, includeMediaGroups, errorFactory, filter, mapper) suspend fun BehaviourContext.waitVideoNoteMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitVoiceMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper) suspend fun BehaviourContext.waitInvoiceMessage( initRequest: Request<*>? = null, errorFactory: NullableRequestBuilder<*> = { null }, count: Int = 1, filter: SimpleFilter>? = null, mapper: CommonMessageToCommonMessageMapper? = null ) = waitContentMessage(count, initRequest, false, errorFactory, filter, mapper)