2023-07-14 19:47:41 +00:00
# db - msg_id, user_id, guild_id
2023-08-01 21:52:03 +00:00
import asyncio
2023-07-14 19:47:41 +00:00
import discord
import os
import json
from dotenv import load_dotenv
from discord . ui import Modal , InputText
2023-07-14 20:57:01 +00:00
from discord . utils import get
2023-08-09 20:54:38 +00:00
from discord . ext import commands
2023-08-08 15:34:51 +00:00
from dbutil import MessageDB
2023-08-10 20:29:21 +00:00
from dbutil import GuildAppDB
2023-07-14 19:47:41 +00:00
load_dotenv ( )
TOKEN = os . getenv ( " TOKEN " )
SERVER_NAME = os . getenv ( " SERVER_NAME " )
2023-07-15 21:32:28 +00:00
CHANNEL_ID = os . getenv ( " CHANNEL_ID " )
2023-07-14 19:47:41 +00:00
bot = discord . Bot ( intents = discord . Intents . all ( ) )
2023-07-15 21:32:28 +00:00
channel_id = int ( CHANNEL_ID )
2023-07-14 19:47:41 +00:00
2023-08-09 20:08:23 +00:00
questions = [
" What is your Minecraft username? " ,
" How did you learn about CraftTopia? (If it was from a friend or a website we would like to know either who or where just to know where credit is due) " ,
" Have you been a part of a minecraft community before, if so, let us know what your experience was with that community. " ,
" How old are you? " ,
" Do you have a good microphone for proximity chat? " ,
" Do you plan on spending at least 2-3 hours a week on the server (our current definition of active) " ,
" What will you be able to do to help us grow and build our community? " ,
" Any other questions or concerns? "
]
2023-07-14 19:47:41 +00:00
max_questions = len ( questions )
@bot.event
async def on_ready ( ) :
bot . add_view ( ApplicationButtonsView ( ) )
2023-07-18 14:02:42 +00:00
bot . add_view ( ApplicationStartButtonView ( ) )
2023-08-09 22:08:45 +00:00
activity = discord . Activity ( name = f " { len ( bot . guilds ) } guilds " , type = discord . ActivityType . listening )
await bot . change_presence ( activity = activity , status = discord . Status . online )
2023-07-14 19:47:41 +00:00
print ( f " Logged in as { bot . user } " )
2023-08-11 20:23:43 +00:00
await bot . sync_commands ( force = True )
2023-08-10 20:29:21 +00:00
for i in bot . guilds :
if str ( i . id ) not in GuildAppDB . get_all_guilds ( ) :
GuildAppDB . create_guild ( str ( i . id ) , i . name )
print ( f " entry for { i . id } created " )
2023-07-14 19:47:41 +00:00
2023-08-09 22:08:45 +00:00
@bot.event
async def on_guild_join ( guild ) :
activity = discord . Activity ( name = f " { len ( bot . guilds ) } guilds " , type = discord . ActivityType . listening )
await bot . change_presence ( activity = activity , status = discord . Status . online )
2023-08-10 20:29:21 +00:00
GuildAppDB . create_guild ( str ( guild . id ) , guild . name )
print ( f " Joined guild { guild . name } : { guild . id } " )
2023-08-09 22:08:45 +00:00
@bot.event
async def on_guild_remove ( guild ) :
activity = discord . Activity ( name = f " { len ( bot . guilds ) } guilds " , type = discord . ActivityType . listening )
await bot . change_presence ( activity = activity , status = discord . Status . online )
2023-08-10 20:29:21 +00:00
print ( f " Removed from guild { guild . name } : { guild . id } " )
2023-08-09 22:08:45 +00:00
2023-08-10 22:24:03 +00:00
@commands.has_permissions ( administrator = True )
@bot.slash_command ( description = " Command used to set up the application prompt " )
async def start_button ( ctx ) :
embed = discord . Embed ( title = " **Start your application!** " )
embed . add_field ( name = " Click the button below to start your application " , value = " " , inline = False )
embed . set_footer ( text = " Made by @anorak01 " , icon_url = " https://cdn.discordapp.com/avatars/269164865480949760/a1af9962da20d5ddaa136043cf45d015?size=1024 " )
appStartView = ApplicationStartButtonView ( )
await ctx . response . send_message ( " Message set up " , ephemeral = True )
await ctx . channel . send ( embeds = [ embed ] , view = appStartView )
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
@start_button.error
async def on_application_command_error ( ctx , error ) :
if isinstance ( error , commands . MissingPermissions ) :
await ctx . respond ( " You need Administrator permissions to use this command " , ephemeral = True )
else :
raise error
2023-07-14 19:47:41 +00:00
2023-08-11 20:23:43 +00:00
application = discord . SlashCommandGroup ( " application " , " The main command to manage applications " )
@application.command ( description = " Create application " )
async def create ( ctx , application ) :
if len ( application ) < 40 :
result = GuildAppDB . add_application_entry ( str ( ctx . guild . id ) , application )
if result == " success " :
await ctx . response . send_message ( f " Successfully created application: { application } " , ephemeral = True ) # create a new application, modal with name ask
else :
print ( result )
else :
await ctx . response . send_message ( f " please choose shorter name " , ephemeral = True )
@application.command ( description = " Remove application " )
async def remove ( ctx , application ) :
result = GuildAppDB . remove_application_entry ( str ( ctx . guild . id ) , application )
if result == " success " :
await ctx . response . send_message ( f " Successfully removed application: { application } " , ephemeral = True )
else :
await ctx . response . send_message ( f " Application { application } not found " )
@application.command ( description = " List all applications " )
async def list ( ctx ) :
applications = GuildAppDB . get_applications ( str ( ctx . guild . id ) )
print ( applications )
embed = discord . Embed ( )
for i in applications :
embed . add_field ( name = i , value = " " , inline = False )
await ctx . response . send_message ( embed = embed )
@application.command ( description = " Opens editor for selected application " )
async def editor ( ctx ) :
view = discord . ui . View ( )
options = SelectApplicationOptionsEditor ( max_values = 1 , placeholder = " Select application " )
for i in GuildAppDB . get_applications ( str ( ctx . guild . id ) ) :
options . add_option ( label = i , value = i )
view . add_item ( options )
await ctx . response . send_message ( view = view , ephemeral = True )
@application.command ( description = " Select response channel for application " )
async def response_channel ( ctx ) :
view = discord . ui . View ( )
options = SelectApplicationOptionsRespChannel ( max_values = 1 , placeholder = " Select application " )
for i in GuildAppDB . get_applications ( str ( ctx . guild . id ) ) :
options . add_option ( label = i , value = i )
view . add_item ( options )
await ctx . response . send_message ( view = view , ephemeral = True )
'''
@application.command ( description = " test " )
async def test ( ctx ) :
view = SelectResponseChannelView ( )
await ctx . response . send_message ( view = view , ephemeral = True )
@application.command ( description = " appl " )
async def appl ( ctx ) :
viewx = discord . ui . View ( )
options = SelectApplicationOptionsEditor ( max_values = 1 )
for i in GuildAppDB . get_applications ( str ( ctx . guild . id ) ) :
options . add_option ( label = i , value = i )
viewx . add_item ( options )
await ctx . response . send_message ( view = viewx , ephemeral = True ) '''
bot . add_application_command ( application ) # add application group commands
class SelectResponseChannelView ( discord . ui . View ) :
@discord.ui.select (
select_type = discord . ComponentType . channel_select ,
channel_types = [ discord . ChannelType . text ] ,
max_values = 1
)
async def select_callback ( self , select , interaction : discord . Interaction ) :
self . disable_all_items ( )
print ( select . values [ 0 ] . id )
GuildAppDB . set_response_channel ( interaction . guild . id , )
await interaction . response . edit_message ( content = f " Selected channel: { select . values [ 0 ] . mention } " , view = None )
class SelectResponseChannel ( discord . ui . Select ) :
def set_app_name ( self , app_name ) :
self . app_name = app_name
async def callback ( self , interaction : discord . Interaction ) :
self . disabled = True
GuildAppDB . set_response_channel ( str ( interaction . guild . id ) , self . app_name , str ( self . values [ 0 ] . id ) )
await interaction . response . edit_message ( content = f " Selected channel: { self . values [ 0 ] . mention } for application: { self . app_name } " , view = None )
class SelectApplicationOptionsEditor ( discord . ui . Select ) :
async def callback ( self , interaction : discord . Interaction ) :
self . disabled = True
editor = ApplicationEditorView ( str ( interaction . guild . id ) , self . values [ 0 ] )
embed = get_questions_embed ( str ( interaction . guild . id ) , self . values [ 0 ] )
await interaction . response . edit_message ( embed = embed , view = editor )
class SelectApplicationOptionsRespChannel ( discord . ui . Select ) :
async def callback ( self , interaction : discord . Interaction ) :
self . disabled = True
view = discord . ui . View ( )
options = SelectResponseChannel ( select_type = discord . ComponentType . channel_select , channel_types = [ discord . ChannelType . text ] , max_values = 1 )
options . set_app_name ( self . values [ 0 ] )
view . add_item ( options )
await interaction . response . edit_message ( view = view )
def get_questions_embed ( guild_id , application ) - > discord . Embed :
embed = discord . Embed ( title = f " Application: { application } " )
questions , length = GuildAppDB . get_questions ( str ( guild_id ) , application )
for i , que in enumerate ( questions ) :
embed . add_field ( value = f " ** { i + 1 } . { que } ** " , name = " " , inline = False )
embed . set_footer ( text = " Made by @anorak01 " , icon_url = " https://cdn.discordapp.com/avatars/269164865480949760/a1af9962da20d5ddaa136043cf45d015?size=1024 " )
return embed
class ApplicationEditorView ( discord . ui . View ) :
def __init__ ( self , guild_id , application_name ) :
super ( ) . __init__ ( timeout = 180 )
self . guild_id = guild_id
self . application_name = application_name
@discord.ui.button (
label = " New " ,
style = discord . ButtonStyle . green ,
custom_id = " editor:add " ,
row = 0
)
async def add_question ( self , button : discord . ui . Button , interaction : discord . Interaction ) :
print ( " add question " )
modal = AddQuestionModal ( self . application_name )
await interaction . response . send_modal ( modal )
@discord.ui.button (
label = " Remove " ,
style = discord . ButtonStyle . red ,
custom_id = " editor:remove " ,
row = 0
)
async def remove_question ( self , button , interaction : discord . Interaction ) :
print ( " remove question " )
view = ApplicationEditorView ( str ( interaction . guild . id ) , self . application_name )
options = RemoveQuestionSelect ( max_values = 1 , placeholder = " Select question to remove " )
options . set_app_name ( self . application_name )
questions , length = GuildAppDB . get_questions ( str ( interaction . guild . id ) , self . application_name )
for i , que in enumerate ( questions ) :
options . add_option ( label = f " { str ( i + 1 ) } . { que } " , value = str ( i ) )
view . add_item ( options )
await interaction . response . edit_message ( view = view )
@discord.ui.button (
label = " Edit " ,
style = discord . ButtonStyle . primary ,
custom_id = " editor:edit " ,
row = 0
)
async def edit_question ( self , button , interaction : discord . Interaction ) :
print ( " edit question " )
view = ApplicationEditorView ( str ( interaction . guild . id ) , self . application_name )
options = EditQuestionSelect ( max_values = 1 , placeholder = " Select question to edit " )
options . set_app_name ( self . application_name )
questions , length = GuildAppDB . get_questions ( str ( interaction . guild . id ) , self . application_name )
for i , que in enumerate ( questions ) :
options . add_option ( label = f " { str ( i + 1 ) } . { que } " , value = str ( i ) )
view . add_item ( options )
await interaction . response . edit_message ( view = view )
@discord.ui.button (
label = " Move " ,
style = discord . ButtonStyle . gray ,
custom_id = " editor:move " ,
row = 0
)
async def move_question ( self , button , interaction : discord . Interaction ) :
print ( " move question " )
view = ApplicationEditorView ( str ( interaction . guild . id ) , self . application_name )
options = MoveQuestionSelect ( max_values = 1 , placeholder = " Select question to move " )
options . set_app_name ( self . application_name )
questions , length = GuildAppDB . get_questions ( str ( interaction . guild . id ) , self . application_name )
for i , que in enumerate ( questions ) :
options . add_option ( label = f " { str ( i + 1 ) } . { que } " , value = str ( i ) )
view . add_item ( options )
await interaction . response . edit_message ( view = view )
class AddQuestionModal ( discord . ui . Modal ) :
def __init__ ( self , app_name ) :
self . app_name = app_name
super ( ) . __init__ ( discord . ui . InputText ( label = f " New Question: " ) , title = " " )
async def callback ( self , interaction : discord . Interaction ) :
question = self . children [ 0 ] . value
GuildAppDB . add_question ( str ( interaction . guild . id ) , self . app_name , question )
embed = get_questions_embed ( str ( interaction . guild . id ) , self . app_name )
await interaction . response . edit_message ( embed = embed )
class RemoveQuestionSelect ( discord . ui . Select ) :
def set_app_name ( self , app_name ) :
self . app_name = app_name
async def callback ( self , interaction : discord . Interaction ) :
self . disabled = True
GuildAppDB . remove_question ( str ( interaction . guild . id ) , self . app_name , int ( self . values [ 0 ] ) + 1 )
editor = ApplicationEditorView ( str ( interaction . guild . id ) , self . app_name )
embed = get_questions_embed ( str ( interaction . guild . id ) , self . app_name )
await interaction . response . edit_message ( embed = embed , view = editor )
class EditQuestionSelect ( discord . ui . Select ) :
def set_app_name ( self , app_name ) :
self . app_name = app_name
async def callback ( self , interaction : discord . Interaction ) :
self . disabled = True
editor = ApplicationEditorView ( str ( interaction . guild . id ) , self . app_name )
modal = EditQuestionModal ( self . app_name , int ( self . values [ 0 ] ) + 1 )
await interaction . response . send_modal ( modal )
await interaction . followup . edit_message ( view = editor , message_id = interaction . message . id )
class EditQuestionModal ( discord . ui . Modal ) :
def __init__ ( self , app_name , question_index ) :
self . app_name = app_name
self . question_index = question_index
super ( ) . __init__ ( discord . ui . InputText ( label = f " Edited Question: " ) , title = " " )
async def callback ( self , interaction : discord . Interaction ) :
question = self . children [ 0 ] . value
GuildAppDB . edit_question ( str ( interaction . guild . id ) , self . app_name , self . question_index , question )
embed = get_questions_embed ( str ( interaction . guild . id ) , self . app_name )
await interaction . response . edit_message ( embed = embed )
class MoveQuestionSelect ( discord . ui . Select ) :
def set_app_name ( self , app_name ) :
self . app_name = app_name
async def callback ( self , interaction : discord . Interaction ) :
self . disabled = True
view = ApplicationEditorView ( str ( interaction . guild . id ) , self . app_name )
options = MoveQuestionSelectNum ( max_values = 1 , placeholder = " Select place for move " )
options . set_app_name ( self . app_name )
options . set_init_index ( int ( self . values [ 0 ] ) + 1 )
questions , length = GuildAppDB . get_questions ( str ( interaction . guild . id ) , self . app_name )
for i in range ( length ) :
options . add_option ( label = str ( i + 1 ) , value = str ( i + 1 ) )
view . add_item ( options )
await interaction . response . edit_message ( view = view )
class MoveQuestionSelectNum ( discord . ui . Select ) :
def set_app_name ( self , app_name ) :
self . app_name = app_name
def set_init_index ( self , init_index : int ) :
self . init_index = init_index
async def callback ( self , interaction : discord . Interaction ) :
self . disabled = True
editor = ApplicationEditorView ( str ( interaction . guild . id ) , self . app_name )
GuildAppDB . move_question ( str ( interaction . guild . id ) , self . app_name , int ( self . init_index ) , int ( self . values [ 0 ] ) )
embed = get_questions_embed ( str ( interaction . guild . id ) , self . app_name )
await interaction . response . edit_message ( view = editor , embed = embed )
2023-08-10 22:24:03 +00:00
# View with button that starts the application process
class ApplicationStartButtonView ( discord . ui . View ) :
def __init__ ( self ) :
super ( ) . __init__ ( timeout = None )
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
@discord.ui.button (
label = " Start application! " ,
style = discord . ButtonStyle . green ,
custom_id = f " persistent:start_application " ,
)
async def start_app ( self , button : discord . ui . Button , interaction : discord . Interaction ) :
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
user = await interaction . user . create_dm ( )
embedd = discord . Embed ( title = f ' CreaTopia Application ' , description = " Hey! Your application has started. You have 300 seconds to complete it. " )
embedd . add_field ( value = f ' You can cancel the application by answering " cancel " to any of the questions ' , name = " " , inline = False )
embedd . set_footer ( text = " Made by @anorak01 " , icon_url = " https://cdn.discordapp.com/avatars/269164865480949760/a1af9962da20d5ddaa136043cf45d015?size=1024 " )
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
try :
await user . send ( embed = embedd )
except discord . Forbidden :
await interaction . response . send_message ( content = " Can ' t start application. Please allow direct messages from server members in your privacy settings. " , ephemeral = True )
return
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
await interaction . response . send_message ( content = " Application started " , ephemeral = True )
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
application = { ' userId ' : interaction . user . id }
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
for i in range ( 0 , max_questions ) :
try :
embed = discord . Embed ( title = f ' Question [ { i + 1 } / { max_questions } ] ' , description = questions [ i ] )
await user . send ( embed = embed )
response = await bot . wait_for ( ' message ' , check = lambda m : m . author == interaction . user and m . channel == user , timeout = 300 )
if response . content . startswith ( " cancel " ) :
await user . send ( " Your application has been cancelled " )
return
else :
application [ f ' question { i } ' ] = response . content
except asyncio . TimeoutError :
await user . send ( content = " As you haven ' t replied in 300 seconds, your application has been cancelled " )
return
try :
with open ( ' applications.json ' , ' r ' ) as f :
data = json . load ( f )
except ( FileNotFoundError , json . JSONDecodeError ) :
data = [ ]
data . append ( application )
with open ( ' applications.json ' , ' w ' ) as f :
json . dump ( data , f )
channel = bot . get_channel ( channel_id )
embed = discord . Embed ( title = ' Application: ' + interaction . user . display_name )
for i in range ( 0 , max_questions ) :
embed . add_field ( name = f ' { questions [ i ] } ' , value = application [ f ' question { i } ' ] , inline = False )
embed . set_footer ( text = f " Applicant ID: { interaction . user . id } " )
appView = ApplicationButtonsView ( )
msg = await channel . send ( embed = embed , view = appView )
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
print ( msg . id )
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
MessageDB . add_application_msg ( msg . id , interaction . user . id , interaction . guild . id )
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
await user . send ( ' Thank you for applying! ' )
2023-07-14 19:47:41 +00:00
2023-08-10 22:24:03 +00:00
# View containing accept and decline buttons for each application
2023-07-14 19:47:41 +00:00
class ApplicationButtonsView ( discord . ui . View ) :
def __init__ ( self ) :
super ( ) . __init__ ( timeout = None )
@discord.ui.button (
label = " Accept " ,
style = discord . ButtonStyle . green ,
custom_id = f " persistent:accept " ,
)
async def accept ( self , button : discord . ui . Button , interaction : discord . Interaction ) :
msg_id = str ( interaction . message . id )
2023-08-08 15:34:51 +00:00
user_id , guild_id = MessageDB . get_application_msg ( msg_id )
2023-08-01 21:54:41 +00:00
2023-07-14 19:47:41 +00:00
modal = ApplicationModal ( title = f " Accepting: { bot . get_user ( user_id ) . display_name } " )
modal . set_action ( " acc " )
modal . add_item ( discord . ui . InputText ( label = f " Reason: " ) )
await interaction . response . send_modal ( modal )
@discord.ui.button (
label = " Decline " ,
style = discord . ButtonStyle . red ,
custom_id = f " persistent:decline " ,
)
async def decline ( self , button : discord . ui . Button , interaction : discord . Interaction ) :
msg_id = str ( interaction . message . id )
2023-08-08 15:34:51 +00:00
user_id , guild_id = MessageDB . get_application_msg ( msg_id )
2023-08-01 21:54:07 +00:00
2023-07-14 19:47:41 +00:00
modal = ApplicationModal ( title = f " Declining: { bot . get_user ( user_id ) . display_name } " )
modal . set_action ( " dec " )
modal . add_item ( discord . ui . InputText ( label = f " Reason: " ) )
await interaction . response . send_modal ( modal )
2023-08-10 22:24:03 +00:00
# Modal functioning as a callback for Accepting/Declining application
2023-07-14 19:47:41 +00:00
class ApplicationModal ( discord . ui . Modal ) :
def set_action ( self , action ) :
self . action = action
async def callback ( self , interaction : discord . Interaction ) :
reason = self . children [ 0 ] . value
msg_id = str ( interaction . message . id )
2023-08-08 15:34:51 +00:00
user_id , guild_id = MessageDB . get_application_msg ( msg_id )
2023-07-14 19:47:41 +00:00
if self . action == " acc " :
user = await bot . get_user ( user_id ) . create_dm ( )
await user . send ( f " You have been accepted to the CreatTopia Minecraft server! " )
await user . send ( f " Reason: { reason } " )
await interaction . response . send_message ( content = " Application accepted " , ephemeral = True )
2023-07-14 20:57:01 +00:00
role = get ( interaction . message . guild . roles , name = " CreatTopian " )
await discord . utils . get ( interaction . message . guild . members , id = int ( user_id ) ) . add_roles ( role )
2023-07-15 21:32:28 +00:00
emb = interaction . message . embeds [ 0 ]
emb . colour = discord . Colour . green ( )
2023-07-16 09:10:19 +00:00
embed = discord . Embed ( title = ' Accepted ' )
embed . add_field ( name = f ' Reason: ' , value = reason , inline = False )
embed . colour = discord . Colour . green ( )
await interaction . followup . edit_message ( message_id = interaction . message . id , embeds = [ emb , embed ] )
2023-08-04 13:40:40 +00:00
view = discord . ui . View . from_message ( interaction . message )
view . disable_all_items ( )
await interaction . followup . edit_message ( message_id = interaction . message . id , view = view )
2023-07-14 19:47:41 +00:00
if self . action == " dec " :
user = await bot . get_user ( user_id ) . create_dm ( )
await user . send ( f " You have been declined access to the CreatTopia Minecraft server. " )
await user . send ( f " Reason: { reason } " )
await interaction . response . send_message ( content = " Application declined " , ephemeral = True )
2023-07-15 21:32:28 +00:00
emb = interaction . message . embeds [ 0 ]
emb . colour = discord . Colour . red ( )
2023-07-16 09:10:19 +00:00
embed = discord . Embed ( title = ' Declined ' )
embed . add_field ( name = f ' Reason: ' , value = reason , inline = False )
embed . colour = discord . Colour . red ( )
await interaction . followup . edit_message ( message_id = interaction . message . id , embeds = [ emb , embed ] )
2023-08-04 13:40:40 +00:00
view = discord . ui . View . from_message ( interaction . message )
view . disable_all_items ( )
await interaction . followup . edit_message ( message_id = interaction . message . id , view = view )
2023-07-14 19:47:41 +00:00
2023-07-17 10:26:28 +00:00
2023-08-10 22:24:03 +00:00
# end
2023-07-14 19:47:41 +00:00
bot . run ( TOKEN )