質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

87.59%

Discord Bot VCに居た時間を測定して表示する

受付中

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 1,085

score 2

目的

discordのボイチャに入ったら、入った時間だけ時間を表示されるbotを作りたい。

言語

python

バージョン・ソフト

python3.9.4

vscode

参考にしたサイト

https://qiita.com/tokkq/items/311aa297175b9cf7f946

コード

import discord
import datetime

client = discord.Client()
pretime_dict = {}

@client.event
async def on_ready():
print('ログインしました')

@client.event
async def on_voice_state_update(member, before, after):
print("ボイスチャンネルで変化がありました")

if((before.self_mute is not after.self_mute) or (before.voice.self_deaf is not after.voice.self_deaf)):
print("ボイスチャンネルでミュート設定の変更がありました")
return

if(before.voice_channel is None):
pretime_dict[after.name] = datetime.datetime.now()
elif(after.voice_channel is None):
duration_time = pretime_dict[before.name] - datetime.datetime.now()
duration_time_adjust = int(duration_time.total_seconds()) * -1

reply_channel_name = "Study-time"
reply_channel = [channel for channel in before.server.channels if channel.name == reply_channel_name][0]
reply_text = after.name + " が "+ before.voice_channel.name + " から抜けました。 通話時間:" + str(duration_time_adjust) +"秒"

await client.send_message(reply_channel ,reply_text)

client.run("token")

追記2

import discord
import datetime

client = discord.Client()
pretime_dict = {}

@client.event
async def on_ready():
    print('ログインしました')

@client.event
async def on_voice_state_update(member, before, after):
    print("ボイスチャンネルで変化がありました")

    if((before.self_mute is not after.self_mute) or (before.voice.self_deaf is not after.voice.self_deaf)):
        print("ボイスチャンネルでミュート設定の変更がありました")
        return

    if(before.voice_channel is None):
        pretime_dict[after.name] = datetime.datetime.now()
    elif(after.voice_channel is None):
        duration_time = pretime_dict[before.name] - datetime.datetime.now()
        duration_time_adjust = int(duration_time.total_seconds()) * -1

    reply_channel_name = "#Study-time"
    reply_channel = [channel for channel in before.server.channels if channel.name == reply_channel_name][0]
    reply_text = after.name + " が "+ before.voice_channel.name + " から抜けました。 通話時間:" + str(duration_time_adjust) +"秒"

    await Client.send_message(reply_channel ,reply_text)

client.run("token")

#エラー ※起動してボイチャに入って抜けて終わる動作です。
ボイスチャンネルで変化がありました
Ignoring exception in on_voice_state_update      
Traceback (most recent call last):
  File "C:\Program Files\Python39\lib\site-packages\discord\client.py", line 343, in _run_event   
    await coro(*args, **kwargs)
  File "c:\Users\user\Desktop\nob\Virtual_currency_of_the_tool\noob.py", line 15, in on_voice_state_update
    if((before.self_mute is not after.self_mute) 
or (before.voice.self_deaf is not after.voice.self_deaf)):
AttributeError: 'VoiceState' object has no attribute 'voice'
ログインしました
ボイスチャンネルで変化がありました
Ignoring exception in on_voice_state_update      
Traceback (most recent call last):
  File "C:\Program Files\Python39\lib\site-packages\discord\client.py", line 343, in _run_event   
    await coro(*args, **kwargs)
  File "c:\Users\user\Desktop\nob\Virtual_currency_of_the_tool\noob.py", line 15, in on_voice_state_update
    if((before.self_mute is not after.self_mute) 
or (before.voice.self_deaf is not after.voice.self_deaf)):
AttributeError: 'VoiceState' object has no attribute 'voice'
Exception ignored in: <function _ProactorBasePipeTransport.__del__ at 0x00000255F0F0ADC0>
Traceback (most recent call last):
  File "C:\Program Files\Python39\lib\asyncio\proactor_events.py", line 116, in __del__
    self.close()ボイスチャンネルで変化がありました
Ignoring exception in on_voice_state_update      
Traceback (most recent call last):
  File "C:\Program Files\Python39\lib\site-packages\discord\client.py", line 343, in _run_event   
    await coro(*args, **kwargs)
  File "c:\Users\user\Desktop\nob\Virtual_currency_of_the_tool\noob.py", line 15, in on_voice_state_update
    if((before.self_mute is not after.self_mute) 
or (before.voice.self_deaf is not after.voice.self_deaf)):
AttributeError: 'VoiceState' object has no attribute 'voice'
ログインしました
ボイスチャンネルで変化がありました
Ignoring exception in on_voice_state_update      
Traceback (most recent call last):
  File "C:\Program Files\Python39\lib\site-packages\discord\client.py", line 343, in _run_event   
    await coro(*args, **kwargs)
  File "c:\Users\user\Desktop\nob\Virtual_currency_of_the_tool\noob.py", line 15, in on_voice_state_update
    if((before.self_mute is not after.self_mute) 
or (before.voice.self_deaf is not after.voice.self_deaf)):
AttributeError: 'VoiceState' object has no attribute 'voice'
Exception ignored in: <function _ProactorBasePipeTransport.__del__ at 0x00000255F0F0ADC0>
Traceback (most recent call last):
  File "C:\Program Files\Python39\lib\asyncio\proactor_events.py", line 116, in __del__
    self.close()
コード
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • munya24

    2021/04/14 15:23

    できました
    インデントです。すいません

    キャンセル

  • K_3578

    2021/04/14 15:25

    謝らなくていいです。あまり卑屈になられてもこちらも困惑するので。

    キャンセル

  • munya24

    2021/04/14 15:46

    はい

    キャンセル

回答 1

+1

voice_state_updateイベントのイベントハンドラには3つの位置引数が渡されるのに before と after しか受け取らないように記述されているからですよ。

@client.event
async def on_voice_state_update(member, before, after):

でどうでしょう。

(追記)

AttributeError: 'VoiceState' object has no attribute 'voice'

参考サイトのコードは投稿日を見る限りかなり古いので参考になりません。
全体的に書き直すことになりそうです。

この AttributeError は、before、afterにはそれぞれVoiceStateオブジェクトが入るのでvoice属性を参照できないために発生します。

before.voice.self_mute などvoice属性を参照する部分は before.self_mute というように置き換えてみてください。


(追記2)

エラーの読み方を知ったほうがよいですね

AttributeError: '〇〇' object has no attribute '▲▲'

このエラー文はそのまま 「▲▲」という属性は 〇〇 オブジェクトにはない、ということです
AttributeError: 'VoiceState' object has no attribute 'voice'

ゆえにこれは VoiceState オブジェクトは voice 属性を持っていないという意味になります。
VoiceState はここでは before と after の型です
どこで voice 属性を参照しているでしょうか?

ここです:

if((before.self_mute is not after.self_mute) or (before.voice.self_deaf is not after.voice.self_deaf)):

before.voice.self_deaf is not after.voice.self_deaf

before や after にvoice属性はないことを念頭に置いてください:

before.self_deaf is not after.self_deaf

(追記3)

Pythonでは、同じレベルのブロックの中ではインデントの数は同じにする必要があります。
というより、インデントの数によってブロックの深さを表現しています。

    print("ボイスチャンネルで変化がありました")

if((before.self_mute is not after.self_mute) or (before.voice.self_deaf is not after.voice.self_deaf)):
    print("ボイスチャンネルでミュート設定の変更がありました")
    return

  if(before.voice_channel is None):
    pretime_dict[after.name] = datetime.datetime.now()

インデントの数がどれもバラバラなので、これはエラーになります。

ブロックというのは、「:」記号以降に記述された一連の処理です。

# ここは一番浅く

def f():
    #  関数宣言には「:」がつくので1段下げる
    for i in range(5):
        #  for文には「:」がつくので1段下げる

このように、「:」でブロックが発生したら1段下げるように覚えればよいと思います
(インデントは半角4個のスペースで表現するのが一般的)

ゆえに、

@client.event
async def on_voice_state_update(member, before, after):
    print("ボイスチャンネルで変化がありました")

    if((before.self_mute is not after.self_mute) or (before.voice.self_deaf is not after.voice.self_deaf)):
        print("ボイスチャンネルでミュート設定の変更がありました")
        return

    if(before.voice_channel is None):
        pretime_dict[after.name] = datetime.datetime.now()
    elif(after.voice_channel is None):
        duration_time = pretime_dict[before.name] - datetime.datetime.now()
        duration_time_adjust = int(duration_time.total_seconds()) * -1

このように修正するとよいです。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2021/04/14 16:47

    追記しました。

    キャンセル

  • 2021/04/14 19:41

    追記しました。

    キャンセル

  • 2021/04/15 08:03

    なぜbeforeやafterからvoice属性を参照したままなのでしょうか?

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 87.59%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る