You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
GPT3Discord/cogs/translation_service_cog.py

149 lines
5.2 KiB

import asyncio
import os
import traceback
import aiohttp
import discord
# We don't use the converser cog here because we want to be able to redo for the last images and text prompts at the same time
from sqlitedict import SqliteDict
from models.deepl_model import TranslationModel
from services.environment_service import EnvService
from services.image_service import ImageService
from services.text_service import TextService
ALLOWED_GUILDS = EnvService.get_allowed_guilds()
def build_translation_embed(text, translated_text, translated_language):
"""Build an embed for the translation"""
embed = discord.Embed(
title=f"Translation to " + translated_language,
color=0x311432,
)
embed.add_field(name="Original text", value=text, inline=False)
embed.add_field(name="Translated Text", value=translated_text, inline=False)
return embed
class TranslationService(discord.Cog, name="TranslationService"):
"""Cog containing a draw commands and file management for saved images"""
def __init__(
self,
bot,
translation_model,
):
super().__init__()
self.bot = bot
self.translation_model = translation_model
# Make a mapping of all the country codes and their full country names:
def build_supported_language_embed(self):
"""Build an embed for the translation"""
embed = discord.Embed(
title=f"Translator supported languages",
color=0x311432,
)
# Add the list of supported languages in a nice format
embed.add_field(
name="Languages",
value=", ".join(
[f"{name}" for name in TranslationModel.get_all_country_names()]
),
inline=False,
)
return embed
async def translate_command(self, ctx, text, target_language, formality):
"""Delete all local images"""
await ctx.defer()
# TODO Add pagination!
if (
target_language.lower().strip()
not in TranslationModel.get_all_country_names(lower=True)
):
await ctx.respond(
f"The language {target_language} is not recognized or supported. Please use `/languages` to see the list of supported languages."
)
return
try:
response = await self.translation_model.send_translate_request(
text,
TranslationModel.get_country_code_from_name(target_language),
formality,
)
except aiohttp.ClientResponseError as e:
await ctx.respond(f"There was an error with the DeepL API: {e.message}")
return
await ctx.respond(
embed=build_translation_embed(text, response, target_language)
)
async def translate_action(self, ctx, message):
await ctx.defer(ephemeral=True)
selection_message = await ctx.respond(
"Select language", ephemeral=True, delete_after=60
)
await selection_message.edit(
view=TranslateView(self.translation_model, message, selection_message)
)
async def languages_command(self, ctx):
"""Show all languages supported for translation"""
await ctx.defer()
await ctx.respond(embed=self.build_supported_language_embed())
class TranslateView(discord.ui.View):
def __init__(self, translation_model, message, selection_message):
super().__init__()
self.translation_model = translation_model
self.message = message
self.selection_message = selection_message
@discord.ui.select( # the decorator that lets you specify the properties of the select menu
placeholder="Language", # the placeholder text that will be displayed if nothing is selected
min_values=1, # the minimum number of values that must be selected by the users
max_values=1, # the maximum number of values that can be selected by the users
options=[ # the list of options from which users can choose, a required field
discord.SelectOption(
label=name,
)
for name in TranslationModel.get_all_country_names()
],
)
async def select_callback(
self, select, interaction
): # the function called when the user is done selecting options
try:
response = await self.translation_model.send_translate_request(
self.message.content,
TranslationModel.get_country_code_from_name(select.values[0]),
)
await self.message.reply(
mention_author=False,
embed=build_translation_embed(
self.message.content, response, select.values[0]
),
)
await self.selection_message.delete()
except aiohttp.ClientResponseError as e:
await interaction.response.send_message(
f"There was an error with the DeepL API: {e.message}",
ephemeral=True,
delete_after=15,
)
return
except Exception as e:
await interaction.response.send_message(
f"There was an error: {e}", ephemeral=True, delete_after=15
)
return