# -*- coding: utf-8 -*-
"""
８個のボールが動き回る
"""

import tkinter as tk
import random


# 個々のボールの情報は専用のクラスに
class Ball:
    r = 20  # ボールの半径（当たり判定に使用）

    # クリックされたところにボールを生成
    def __init__(self, event:tk.Event):
        self.x = event.x
        self.y = event.y
        self.canvas:tk.Canvas = event.widget
        # 移動速度は乱数
        self.dx = random.random()*20-10
        self.dy = random.random()*20-10
        # x,y を中心にするため，±rで円を描く
        self.id = self.canvas.create_oval(self.x-self.r, self.y-self.r, self.x+self.r, self.y+self.r)
        # 動作開始
        self.move()

    # 動き回らせる
    # まとめて扱ってもよいが，こちらの方がオブジェクト指向的
    def move(self):
        # 横方向の移動．端に行ったら移動を反転＋乱数
        self.x += self.dx
        if   self.x-self.r < 0:
            self.dx =  abs(self.dx) + random.random()*5-2
        elif self.x+self.r > self.canvas.winfo_width():
            self.dx = -abs(self.dx) + random.random()*5-2

        """ 指示ここから """
        # このコメントブロックの下に，
        # 縦方向の移動．端に行ったら移動を反転＋乱数
        # という動きになるよう，命令文を書きなさい．
        # なお，キャンバスの縦幅は .winfo_height() で取得できる．

        self.y += self.dy
        if   self.y-self.r < 0:
            self.dy =  abs(self.dy) + random.random()*5-2
        elif self.y+self.r > self.canvas.winfo_height():
            self.dy = -abs(self.dy) + random.random()*5-2
        """ 指示ここまで """

        # 移動を反映し，次を呼び出し
        self.canvas.moveto(self.id, self.x-self.r, self.y-self.r)
        self.canvas.after(50, self.move)
    
    # ボールの色を変える
    def change_fill(self, color):
        self.canvas.itemconfig(self.id, fill=color)


# ボール全体の情報をキャンバスに持たせる
class CanvasBox(tk.Canvas):

    # キャンバスの初期設定
    def __init__(self, master):
        super().__init__(master, width=400, height=300)
        self.ball_id:list[Ball] = []
        self.bind("<Button>", self.add_ball)
        self.wrap_ball()

    # ボールを追加するが，８個を超えたら古いものを消す
    def add_ball(self, event:tk.Event):
        self.ball_id.append(Ball(event))
        if len(self.ball_id) > 8:
            # list型データに .pop(0) をすると
            # 先頭要素が取り出され，かつ，list内部から消える
            self.delete(self.ball_id.pop(0).id)
    
    # ボールは黒色だが，重なると赤色にする
    # 個々のボールで行ってもよいが，個から全体を把握するのは難しい
    def wrap_ball(self):
        for ball1 in self.ball_id:
            color = "black"  # デフォルトは黒色
            for ball2 in self.ball_id:
                # 自分以外のボールとの二乗距離が一定以下なら赤色
                if ball1 != ball2:
                    dx2 = (ball1.x - ball2.x)**2
                    dy2 = (ball1.y - ball2.y)**2
                    # ball1もball2も半径が20なので，二乗距離は1600
                    if (dx2 + dy2) < (ball1.r + ball2.r)**2:
                        color = "red"
                        break
            ball1.change_fill(color)
        
        # 次を呼び出し
        self.after(50, self.wrap_ball)



def main():
    base = tk.Tk()
    base.title("$u_id")

    canvas = CanvasBox(base)
    canvas.pack()

    base.mainloop()  # GUIを待機


if __name__ == "__main__":
    main()
