mirror of
https://github.com/InsanusMokrassar/PlaguPoster.git
synced 2024-10-31 21:33:46 +00:00
commit
2b41082a48
@ -10,5 +10,5 @@ android.enableJetifier=true
|
|||||||
# Project data
|
# Project data
|
||||||
|
|
||||||
group=dev.inmo
|
group=dev.inmo
|
||||||
version=0.0.6
|
version=0.0.7
|
||||||
android_code_version=6
|
android_code_version=7
|
||||||
|
@ -1,15 +1,15 @@
|
|||||||
[versions]
|
[versions]
|
||||||
|
|
||||||
kotlin = "1.7.21"
|
kotlin = "1.7.22"
|
||||||
kotlin-serialization = "1.4.1"
|
kotlin-serialization = "1.4.1"
|
||||||
|
|
||||||
plagubot = "3.1.3"
|
plagubot = "3.2.0"
|
||||||
tgbotapi = "4.1.2"
|
tgbotapi = "4.2.1"
|
||||||
microutils = "0.14.2"
|
microutils = "0.16.0"
|
||||||
kslog = "0.5.3"
|
kslog = "0.5.4"
|
||||||
krontab = "0.8.3"
|
krontab = "0.8.4"
|
||||||
tgbotapi-libraries = "0.6.3"
|
tgbotapi-libraries = "0.6.5"
|
||||||
plagubot-plugins = "0.6.1"
|
plagubot-plugins = "0.6.4"
|
||||||
|
|
||||||
dokka = "1.7.20"
|
dokka = "1.7.20"
|
||||||
|
|
||||||
|
2
gradle/wrapper/gradle-wrapper.properties
vendored
2
gradle/wrapper/gradle-wrapper.properties
vendored
@ -1,5 +1,5 @@
|
|||||||
distributionBase=GRADLE_USER_HOME
|
distributionBase=GRADLE_USER_HOME
|
||||||
distributionPath=wrapper/dists
|
distributionPath=wrapper/dists
|
||||||
distributionUrl=https\://services.gradle.org/distributions/gradle-7.5.1-bin.zip
|
distributionUrl=https\://services.gradle.org/distributions/gradle-7.6-bin.zip
|
||||||
zipStoreBase=GRADLE_USER_HOME
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
zipStorePath=wrapper/dists
|
zipStorePath=wrapper/dists
|
||||||
|
@ -10,4 +10,5 @@ import dev.inmo.tgbotapi.types.MessageIdentifier
|
|||||||
interface ReadPostsRepo : ReadCRUDRepo<RegisteredPost, PostId> {
|
interface ReadPostsRepo : ReadCRUDRepo<RegisteredPost, PostId> {
|
||||||
suspend fun getIdByChatAndMessage(chatId: IdChatIdentifier, messageId: MessageIdentifier): PostId?
|
suspend fun getIdByChatAndMessage(chatId: IdChatIdentifier, messageId: MessageIdentifier): PostId?
|
||||||
suspend fun getPostCreationTime(postId: PostId): DateTime?
|
suspend fun getPostCreationTime(postId: PostId): DateTime?
|
||||||
|
suspend fun getFirstMessageInfo(postId: PostId): PostContentInfo?
|
||||||
}
|
}
|
||||||
|
@ -38,9 +38,11 @@ class ExposedPostsRepo(
|
|||||||
|
|
||||||
override val selectById: ISqlExpressionBuilder.(PostId) -> Op<Boolean> = { idColumn.eq(it.string) }
|
override val selectById: ISqlExpressionBuilder.(PostId) -> Op<Boolean> = { idColumn.eq(it.string) }
|
||||||
override val selectByIds: ISqlExpressionBuilder.(List<PostId>) -> Op<Boolean> = { idColumn.inList(it.map { it.string }) }
|
override val selectByIds: ISqlExpressionBuilder.(List<PostId>) -> Op<Boolean> = { idColumn.inList(it.map { it.string }) }
|
||||||
|
override val ResultRow.asId: PostId
|
||||||
|
get() = PostId(get(idColumn))
|
||||||
override val ResultRow.asObject: RegisteredPost
|
override val ResultRow.asObject: RegisteredPost
|
||||||
get() {
|
get() {
|
||||||
val id = PostId(get(idColumn))
|
val id = asId
|
||||||
return RegisteredPost(
|
return RegisteredPost(
|
||||||
id,
|
id,
|
||||||
DateTime(get(createdColumn)),
|
DateTime(get(createdColumn)),
|
||||||
@ -163,4 +165,10 @@ class ExposedPostsRepo(
|
|||||||
override suspend fun getPostCreationTime(postId: PostId): DateTime? = transaction(database) {
|
override suspend fun getPostCreationTime(postId: PostId): DateTime? = transaction(database) {
|
||||||
select { selectById(postId) }.limit(1).firstOrNull() ?.get(createdColumn) ?.let(::DateTime)
|
select { selectById(postId) }.limit(1).firstOrNull() ?.get(createdColumn) ?.let(::DateTime)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override suspend fun getFirstMessageInfo(postId: PostId): PostContentInfo? = transaction(database) {
|
||||||
|
with(contentRepo) {
|
||||||
|
select { postIdColumn.eq(postId.string) }.limit(1).firstOrNull() ?.asObject
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
160
ratings/source/src/commonMain/kotlin/buttons/RootButtons.kt
Normal file
160
ratings/source/src/commonMain/kotlin/buttons/RootButtons.kt
Normal file
@ -0,0 +1,160 @@
|
|||||||
|
package dev.inmo.plaguposter.ratings.source.buttons
|
||||||
|
|
||||||
|
import com.soywiz.klock.DateFormat
|
||||||
|
import dev.inmo.micro_utils.coroutines.runCatchingSafely
|
||||||
|
import dev.inmo.micro_utils.pagination.FirstPagePagination
|
||||||
|
import dev.inmo.micro_utils.pagination.Pagination
|
||||||
|
import dev.inmo.micro_utils.pagination.SimplePagination
|
||||||
|
import dev.inmo.micro_utils.pagination.utils.paginate
|
||||||
|
import dev.inmo.plaguposter.posts.repo.ReadPostsRepo
|
||||||
|
import dev.inmo.plaguposter.ratings.models.Rating
|
||||||
|
import dev.inmo.plaguposter.ratings.repo.RatingsRepo
|
||||||
|
import dev.inmo.plaguposter.ratings.utils.postsByRatings
|
||||||
|
import dev.inmo.tgbotapi.extensions.api.answers.answer
|
||||||
|
import dev.inmo.tgbotapi.extensions.api.edit.edit
|
||||||
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContext
|
||||||
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling.onMessageDataCallbackQuery
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.formatting.makeLinkToMessage
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.types.buttons.dataButton
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.types.buttons.inlineKeyboard
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.types.buttons.urlButton
|
||||||
|
import dev.inmo.tgbotapi.types.ChatIdentifier
|
||||||
|
import dev.inmo.tgbotapi.types.buttons.InlineKeyboardMarkup
|
||||||
|
import dev.inmo.tgbotapi.utils.row
|
||||||
|
|
||||||
|
const val RootButtonsShowRatingData = "ratings_buttons_show"
|
||||||
|
const val RootButtonsShowRatingPageData = "ratings_buttons_show_page"
|
||||||
|
const val RootButtonsToPageData = "ratings_buttons_to_page"
|
||||||
|
|
||||||
|
suspend fun RatingsRepo.buildRootButtons(
|
||||||
|
pagination: Pagination = FirstPagePagination(16),
|
||||||
|
rowSize: Int = 4
|
||||||
|
): InlineKeyboardMarkup {
|
||||||
|
val postsByRatings = postsByRatings().toList().paginate(pagination)
|
||||||
|
return inlineKeyboard {
|
||||||
|
if (postsByRatings.pagesNumber > 1) {
|
||||||
|
row {
|
||||||
|
if (postsByRatings.page > 0) {
|
||||||
|
dataButton("<", "$RootButtonsToPageData ${postsByRatings.page - 1} ${postsByRatings.size}")
|
||||||
|
}
|
||||||
|
dataButton("${postsByRatings.page}: \uD83D\uDD04", "$RootButtonsToPageData ${postsByRatings.page} ${postsByRatings.size}")
|
||||||
|
if (postsByRatings.pagesNumber - postsByRatings.page > 1) {
|
||||||
|
dataButton(">", "$RootButtonsToPageData ${postsByRatings.page + 1} ${postsByRatings.size}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
postsByRatings.results.chunked(rowSize).map {
|
||||||
|
row {
|
||||||
|
it.forEach { (rating, posts) ->
|
||||||
|
dataButton("${rating.double}: ${posts.size}", "$RootButtonsShowRatingData ${rating.double}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val defaultPostCreationTimeFormat: DateFormat = DateFormat("dd.MM.yy HH:mm")
|
||||||
|
|
||||||
|
suspend fun RatingsRepo.buildRatingButtons(
|
||||||
|
postsRepo: ReadPostsRepo,
|
||||||
|
rating: Rating,
|
||||||
|
pagination: Pagination = FirstPagePagination(8),
|
||||||
|
rowSize: Int = 2,
|
||||||
|
postCreationTimeFormat: DateFormat = defaultPostCreationTimeFormat
|
||||||
|
): InlineKeyboardMarkup {
|
||||||
|
val postsByRatings = getPosts(rating .. rating, true).keys.paginate(pagination)
|
||||||
|
return inlineKeyboard {
|
||||||
|
if (postsByRatings.pagesNumber > 1) {
|
||||||
|
row {
|
||||||
|
if (postsByRatings.page > 0) {
|
||||||
|
dataButton("<", "$RootButtonsShowRatingPageData ${postsByRatings.page - 1} ${postsByRatings.size} ${rating.double}")
|
||||||
|
}
|
||||||
|
dataButton("${postsByRatings.page}: \uD83D\uDD04", "$RootButtonsShowRatingPageData ${postsByRatings.page} ${postsByRatings.size} ${rating.double}")
|
||||||
|
if (postsByRatings.pagesNumber - postsByRatings.page > 1) {
|
||||||
|
dataButton(">", "$RootButtonsShowRatingPageData ${postsByRatings.page + 1} ${postsByRatings.size} ${rating.double}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
postsByRatings.results.chunked(rowSize).map {
|
||||||
|
row {
|
||||||
|
it.forEach { postId ->
|
||||||
|
val firstMessageInfo = postsRepo.getFirstMessageInfo(postId) ?: return@forEach
|
||||||
|
val postCreationTime = postsRepo.getPostCreationTime(postId) ?: return@forEach
|
||||||
|
urlButton(
|
||||||
|
postCreationTime.format(postCreationTimeFormat),
|
||||||
|
makeLinkToMessage(
|
||||||
|
firstMessageInfo.chatId,
|
||||||
|
firstMessageInfo.messageId
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
row {
|
||||||
|
dataButton("↩️", "$RootButtonsToPageData 0 16")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.includeRootNavigationButtonsHandler(
|
||||||
|
allowedChats: Set<ChatIdentifier>,
|
||||||
|
ratingsRepo: RatingsRepo,
|
||||||
|
postsRepo: ReadPostsRepo
|
||||||
|
) {
|
||||||
|
suspend fun registerPageQueryListener(
|
||||||
|
dataPrefix: String,
|
||||||
|
onPageUpdate: suspend (pagination: Pagination, additionalParams: Array<String>) -> InlineKeyboardMarkup?
|
||||||
|
) {
|
||||||
|
onMessageDataCallbackQuery(
|
||||||
|
initialFilter = { it.message.chat.id in allowedChats }
|
||||||
|
) {
|
||||||
|
val args = it.data.split(" ").takeIf { it.size >= 3 } ?: return@onMessageDataCallbackQuery
|
||||||
|
val (prefix, pageRaw, sizeRaw) = args
|
||||||
|
|
||||||
|
if (prefix == dataPrefix) {
|
||||||
|
runCatchingSafely {
|
||||||
|
val page = pageRaw.toIntOrNull() ?: return@runCatchingSafely
|
||||||
|
val size = sizeRaw.toIntOrNull() ?: return@runCatchingSafely
|
||||||
|
|
||||||
|
edit(
|
||||||
|
it.message,
|
||||||
|
onPageUpdate(SimplePagination(page, size), args.drop(3).toTypedArray()) ?: return@runCatchingSafely
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
answer(it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
suspend fun registerPageQueryListener(
|
||||||
|
dataPrefix: String,
|
||||||
|
onPageUpdate: suspend (pagination: Pagination) -> InlineKeyboardMarkup?
|
||||||
|
) = registerPageQueryListener(dataPrefix) { pagination, _ ->
|
||||||
|
onPageUpdate(pagination)
|
||||||
|
}
|
||||||
|
registerPageQueryListener(
|
||||||
|
RootButtonsToPageData,
|
||||||
|
ratingsRepo::buildRootButtons
|
||||||
|
)
|
||||||
|
registerPageQueryListener(
|
||||||
|
RootButtonsShowRatingPageData
|
||||||
|
) { pagination, params ->
|
||||||
|
params.firstOrNull() ?.toDoubleOrNull() ?.let { rating ->
|
||||||
|
ratingsRepo.buildRatingButtons(postsRepo, Rating(rating), pagination)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onMessageDataCallbackQuery(
|
||||||
|
initialFilter = { it.message.chat.id in allowedChats }
|
||||||
|
) {
|
||||||
|
val (prefix, ratingRaw) = it.data.split(" ").takeIf { it.size == 2 } ?: return@onMessageDataCallbackQuery
|
||||||
|
|
||||||
|
if (prefix == RootButtonsShowRatingData) {
|
||||||
|
runCatchingSafely {
|
||||||
|
val rating = ratingRaw.toDoubleOrNull() ?: return@runCatchingSafely
|
||||||
|
edit(it.message, ratingsRepo.buildRatingButtons(postsRepo, Rating(rating)))
|
||||||
|
}
|
||||||
|
|
||||||
|
answer(it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -20,6 +20,8 @@ import dev.inmo.plaguposter.posts.panel.PanelButtonsAPI
|
|||||||
import dev.inmo.plaguposter.posts.repo.PostsRepo
|
import dev.inmo.plaguposter.posts.repo.PostsRepo
|
||||||
import dev.inmo.plaguposter.ratings.models.Rating
|
import dev.inmo.plaguposter.ratings.models.Rating
|
||||||
import dev.inmo.plaguposter.ratings.repo.RatingsRepo
|
import dev.inmo.plaguposter.ratings.repo.RatingsRepo
|
||||||
|
import dev.inmo.plaguposter.ratings.source.buttons.buildRootButtons
|
||||||
|
import dev.inmo.plaguposter.ratings.source.buttons.includeRootNavigationButtonsHandler
|
||||||
import dev.inmo.plaguposter.ratings.source.models.*
|
import dev.inmo.plaguposter.ratings.source.models.*
|
||||||
import dev.inmo.plaguposter.ratings.source.repos.*
|
import dev.inmo.plaguposter.ratings.source.repos.*
|
||||||
import dev.inmo.plaguposter.ratings.utils.postsByRatings
|
import dev.inmo.plaguposter.ratings.utils.postsByRatings
|
||||||
@ -34,6 +36,7 @@ import dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling.*
|
|||||||
import dev.inmo.tgbotapi.extensions.utils.extensions.sameMessage
|
import dev.inmo.tgbotapi.extensions.utils.extensions.sameMessage
|
||||||
import dev.inmo.tgbotapi.extensions.utils.types.buttons.dataButton
|
import dev.inmo.tgbotapi.extensions.utils.types.buttons.dataButton
|
||||||
import dev.inmo.tgbotapi.extensions.utils.types.buttons.flatInlineKeyboard
|
import dev.inmo.tgbotapi.extensions.utils.types.buttons.flatInlineKeyboard
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.types.buttons.inlineKeyboard
|
||||||
import dev.inmo.tgbotapi.types.buttons.InlineKeyboardButtons.CallbackDataInlineKeyboardButton
|
import dev.inmo.tgbotapi.types.buttons.InlineKeyboardButtons.CallbackDataInlineKeyboardButton
|
||||||
import dev.inmo.tgbotapi.types.message.textsources.bold
|
import dev.inmo.tgbotapi.types.message.textsources.bold
|
||||||
import dev.inmo.tgbotapi.types.message.textsources.regular
|
import dev.inmo.tgbotapi.types.message.textsources.regular
|
||||||
@ -225,13 +228,23 @@ object Plugin : Plugin {
|
|||||||
+ "• " + bold("% 3.1f".format(it.first.double)) + ": " + bold(it.second.size.toString()) + "\n"
|
+ "• " + bold("% 3.1f".format(it.first.double)) + ": " + bold(it.second.size.toString()) + "\n"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
val keyboard = flatInlineKeyboard {
|
||||||
|
dataButton("Interactive mode", "ratings_interactive")
|
||||||
|
}
|
||||||
runCatchingSafely {
|
runCatchingSafely {
|
||||||
edit(it, textSources)
|
edit(it, textSources, replyMarkup = keyboard)
|
||||||
}.onFailure { _ ->
|
}.onFailure { _ ->
|
||||||
reply(it, textSources)
|
reply(it, textSources, replyMarkup = keyboard)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
includeRootNavigationButtonsHandler(setOf(chatConfig.sourceChatId), ratingsRepo, postsRepo)
|
||||||
|
onMessageDataCallbackQuery("ratings_interactive", initialFilter = { it.message.chat.id == chatConfig.sourceChatId }) {
|
||||||
|
edit(
|
||||||
|
it.message,
|
||||||
|
ratingsRepo.buildRootButtons()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
koin.getOrNull<InlineTemplatesRepo>() ?.apply {
|
koin.getOrNull<InlineTemplatesRepo>() ?.apply {
|
||||||
addTemplate(
|
addTemplate(
|
||||||
|
Loading…
Reference in New Issue
Block a user