如何在 Toplevel Tkinter 中创建形状
How to create shapes in Toplevel Tkinter
每当我尝试在 tkinter Toplevel 中创建形状时,它都会给出此错误消息:
Traceback (most recent call last):
File "C:\Users\tkinter\__init__.py", line 1705, in __call__
return self.func(*args)
File "C:/Users/filea.py", line 183, in nxt_lvl
paddle = Paddle(canvas, 'blue')
File "C:/Users/filea.py", line 125, in __init__
self.id = canvas.create_rectangle(0, 0, 100, 10, fill=color)
AttributeError: 'Toplevel' object has no attribute 'create_rectangle'
这是代码您只需要关注 Ball
Class 和 nxt_lvl
函数:
import random
import time
tk = Tk()
jump_counter = 0
You_win = Label(tk, text="You win!")
class Ball:
def __init__(self, canvas, jumps, paddle, color):
self.canvas = canvas
self.paddle = paddle
self.canvas_height = self.canvas.winfo_height()
print(self.canvas_height)
self.canvas_width = self.canvas.winfo_width()
self.id = canvas.create_oval(10, 10, 25, 25, fill=color)
y = self.id
self.canvas.move(self.id, 245, 100)
self.x = 0
self.y = 25
self.jumps = jumps
self.win = False
self.gravity = 0.1
self.canvas.bind_all('<KeyPress-Up>', self.turn_Up)
self.canvas.bind_all('<KeyRelease-Up>', self.turn_Up_stop)
self.canvas.bind_all('<KeyPress-Left>', self.turn_left)
self.canvas.bind_all('<KeyRelease-Left>', self.turn_left_stop)
self.canvas.bind_all('<KeyPress-Right>', self.turn_right)
self.canvas.bind_all('<KeyRelease-Right>', self.turn_right_stop)
def draw(self):
global jump_counter
paddle_pos = self.canvas.coords(self.paddle.id)
paddle_p2s = self.canvas.coords(self.paddle.id2)
paddle_p3s = self.canvas.coords(self.paddle.id3)
self.canvas.move(self.id, self.x, self.y)
pos = self.canvas.coords(self.id)
self.y += self.gravity
# pos[0] = left side of the Ball
# pos[2] = Right side of the ball
if pos[1] <= 0: # Top of the Ball
You_win.pack()
self.win = True
self.y = 3
if pos[3] >= self.canvas_height: # Bottom of the Ball
jump_counter = 0
self.y = 0
if pos[0] <= 0: # Left Side of the Ball
self.x = 1
if pos[2] >= self.canvas_width: # Right Side of the Ball
self.x = -1
# --------------------------------Hit Collision--------------------------------#
if pos[2] >= paddle_pos[0] and pos[0] <= paddle_pos[2]:
if paddle_pos[1] <= pos[3] <= paddle_pos[3]:
jump_counter = 0
self.y = 0
if pos[2] >= paddle_p2s[0] and pos[0] <= paddle_p2s[2]:
if paddle_p2s[1] <= pos[3] <= paddle_p2s[3]:
jump_counter = 0
self.y = 0
if pos[2] >= paddle_p3s[0] and pos[0] <= paddle_p3s[2]:
if paddle_p3s[1] <= pos[3] <= paddle_p3s[3]:
jump_counter = 0
self.y = 0
if pos[2] <= paddle_pos[2] and pos[0] >= paddle_pos[0]:
if paddle_pos[1] <= pos[1] <= paddle_pos[3]:
self.y = 1
if pos[2] <= paddle_p2s[2] and pos[0] >= paddle_p2s[0]:
if paddle_p2s[1] <= pos[1] <= paddle_p2s[3]:
self.y = 1
if pos[2] <= paddle_p3s[2] and pos[0] >= paddle_p3s[0]:
if paddle_p3s[1] <= pos[1] <= paddle_p3s[3]:
self.y = 1
# If the top of the ball is greater than or equal too paddle's top and if the top of the ball is less than or equal too paddle's bottom
# --------------------------------Hit Collision: End--------------------------------#
def turn_Up(self, evt):
global jump_counter
if jump_counter == self.jumps:
print(jump_counter)
self.y += self.gravity
elif jump_counter != self.jumps:
print('Jump', jump_counter)
jump_counter = jump_counter + 1
self.y = -4
def turn_Up_stop(self, evt):
self.y += self.gravity
def turn_left(self, evt):
self.x = -3
def turn_right(self, evt):
self.x = 3
def turn_left_stop(self, evt):
self.x = self.x
time.sleep(0.005)
self.x = 0
def turn_right_stop(self, evt):
self.x = self.x
time.sleep(0.005)
self.x = 0
class Paddle:
def __init__(self, canvas, color):
pstartsX = [100, 300, 400]
pstartsY = [125, 215, 300]
random.shuffle(pstartsX)
random.shuffle(pstartsY)
self.canvas = canvas
self.id = canvas.create_rectangle(0, 0, 100, 10, fill=color)
self.id2 = canvas.create_rectangle(0, 0, 100, 10, fill=color)
self.id3 = canvas.create_rectangle(0, 0, 100, 10, fill=color)
self.canvas.move(self.id, pstartsX[0], pstartsY[0])
pstartsX.pop(0)
pstartsY.pop(0)
random.shuffle(pstartsX)
random.shuffle(pstartsY)
self.canvas.move(self.id2, pstartsX[0], pstartsY[0])
pstartsX.pop(0)
pstartsY.pop(0)
random.shuffle(pstartsX)
random.shuffle(pstartsY)
self.canvas.move(self.id3, pstartsX[0], pstartsY[0])
pstartsX.pop(0)
pstartsY.pop(0)
random.shuffle(pstartsX)
random.shuffle(pstartsY)
self.y = 0
self.x = 0
self.canvas_width = self.canvas.winfo_width()
def draw(self):
self.canvas.move(self.id, self.x, 0)
pos = self.canvas.coords(self.id)
if pos[0] <= 0:
self.x = 1
elif pos[2] >= self.canvas_width:
self.x = -1
tkwidth = 500
tkheight = 400
tk.title('Level 1')
tk.resizable(0, 0)
tk.wm_attributes("-topmost", 1)
canvas = Canvas(tk, width=tkwidth, height=tkheight, bd=0, highlightthickness=0)
canvas.pack()
tk.update()
paddle = Paddle(canvas, 'blue')
ball = Ball(canvas, 3, paddle, 'green')
print(tkwidth, tkheight)
# width=tkwidth + 100, height=tkheight + 100, bd=0, highlightthickness=0
def nxt_lvl():
global pstartsX
global pstartsY
pstartsX = [100, 300, 400]
pstartsY = [125, 250, 375]
lvl2 = Toplevel(width=tkwidth + 100, height=tkheight + 100, bd=0, highlightthickness=0)
lvl2.title('Level 2')
paddle = Paddle(lvl2, 'blue')
ball = Ball(lvl2, 3, paddle, 'green')
ball.draw()
lvl_2_btn = Button(tk, text="Next Level?", command=nxt_lvl)
while 1:
if not ball.win:
ball.draw()
elif ball.win:
lvl_2_btn.pack()
tk.update_idletasks()
tk.update()
time.sleep(0.01)
考虑这段代码:
lvl2 = Toplevel(width=tkwidth + 100, height=tkheight + 100, bd=0, highlightthickness=0)
lvl2.title('Level 2')
paddle = Paddle(lvl2, 'blue')
ball = Ball(lvl2, 3, paddle, 'green')
请注意,您正在将 lvl2
传递给 Ball
和 Paddle
,而他们期望的位置是 canvas。您需要在 Toplevel
中创建一个 canvas,并将 canvas 传递给 Ball
和 Paddle
.
它应该看起来像这样:
lvl2 = Toplevel(width=tkwidth + 100, height=tkheight + 100, bd=0, highlightthickness=0)
lvl2.title('Level 2')
canvas = Canvas(lvl2)
canvas.pack(fill="both", expand=True)
paddle = Paddle(canvas, 'blue')
ball = Ball(canvas, 3, paddle, 'green')
每当我尝试在 tkinter Toplevel 中创建形状时,它都会给出此错误消息:
Traceback (most recent call last):
File "C:\Users\tkinter\__init__.py", line 1705, in __call__
return self.func(*args)
File "C:/Users/filea.py", line 183, in nxt_lvl
paddle = Paddle(canvas, 'blue')
File "C:/Users/filea.py", line 125, in __init__
self.id = canvas.create_rectangle(0, 0, 100, 10, fill=color)
AttributeError: 'Toplevel' object has no attribute 'create_rectangle'
这是代码您只需要关注 Ball
Class 和 nxt_lvl
函数:
import random
import time
tk = Tk()
jump_counter = 0
You_win = Label(tk, text="You win!")
class Ball:
def __init__(self, canvas, jumps, paddle, color):
self.canvas = canvas
self.paddle = paddle
self.canvas_height = self.canvas.winfo_height()
print(self.canvas_height)
self.canvas_width = self.canvas.winfo_width()
self.id = canvas.create_oval(10, 10, 25, 25, fill=color)
y = self.id
self.canvas.move(self.id, 245, 100)
self.x = 0
self.y = 25
self.jumps = jumps
self.win = False
self.gravity = 0.1
self.canvas.bind_all('<KeyPress-Up>', self.turn_Up)
self.canvas.bind_all('<KeyRelease-Up>', self.turn_Up_stop)
self.canvas.bind_all('<KeyPress-Left>', self.turn_left)
self.canvas.bind_all('<KeyRelease-Left>', self.turn_left_stop)
self.canvas.bind_all('<KeyPress-Right>', self.turn_right)
self.canvas.bind_all('<KeyRelease-Right>', self.turn_right_stop)
def draw(self):
global jump_counter
paddle_pos = self.canvas.coords(self.paddle.id)
paddle_p2s = self.canvas.coords(self.paddle.id2)
paddle_p3s = self.canvas.coords(self.paddle.id3)
self.canvas.move(self.id, self.x, self.y)
pos = self.canvas.coords(self.id)
self.y += self.gravity
# pos[0] = left side of the Ball
# pos[2] = Right side of the ball
if pos[1] <= 0: # Top of the Ball
You_win.pack()
self.win = True
self.y = 3
if pos[3] >= self.canvas_height: # Bottom of the Ball
jump_counter = 0
self.y = 0
if pos[0] <= 0: # Left Side of the Ball
self.x = 1
if pos[2] >= self.canvas_width: # Right Side of the Ball
self.x = -1
# --------------------------------Hit Collision--------------------------------#
if pos[2] >= paddle_pos[0] and pos[0] <= paddle_pos[2]:
if paddle_pos[1] <= pos[3] <= paddle_pos[3]:
jump_counter = 0
self.y = 0
if pos[2] >= paddle_p2s[0] and pos[0] <= paddle_p2s[2]:
if paddle_p2s[1] <= pos[3] <= paddle_p2s[3]:
jump_counter = 0
self.y = 0
if pos[2] >= paddle_p3s[0] and pos[0] <= paddle_p3s[2]:
if paddle_p3s[1] <= pos[3] <= paddle_p3s[3]:
jump_counter = 0
self.y = 0
if pos[2] <= paddle_pos[2] and pos[0] >= paddle_pos[0]:
if paddle_pos[1] <= pos[1] <= paddle_pos[3]:
self.y = 1
if pos[2] <= paddle_p2s[2] and pos[0] >= paddle_p2s[0]:
if paddle_p2s[1] <= pos[1] <= paddle_p2s[3]:
self.y = 1
if pos[2] <= paddle_p3s[2] and pos[0] >= paddle_p3s[0]:
if paddle_p3s[1] <= pos[1] <= paddle_p3s[3]:
self.y = 1
# If the top of the ball is greater than or equal too paddle's top and if the top of the ball is less than or equal too paddle's bottom
# --------------------------------Hit Collision: End--------------------------------#
def turn_Up(self, evt):
global jump_counter
if jump_counter == self.jumps:
print(jump_counter)
self.y += self.gravity
elif jump_counter != self.jumps:
print('Jump', jump_counter)
jump_counter = jump_counter + 1
self.y = -4
def turn_Up_stop(self, evt):
self.y += self.gravity
def turn_left(self, evt):
self.x = -3
def turn_right(self, evt):
self.x = 3
def turn_left_stop(self, evt):
self.x = self.x
time.sleep(0.005)
self.x = 0
def turn_right_stop(self, evt):
self.x = self.x
time.sleep(0.005)
self.x = 0
class Paddle:
def __init__(self, canvas, color):
pstartsX = [100, 300, 400]
pstartsY = [125, 215, 300]
random.shuffle(pstartsX)
random.shuffle(pstartsY)
self.canvas = canvas
self.id = canvas.create_rectangle(0, 0, 100, 10, fill=color)
self.id2 = canvas.create_rectangle(0, 0, 100, 10, fill=color)
self.id3 = canvas.create_rectangle(0, 0, 100, 10, fill=color)
self.canvas.move(self.id, pstartsX[0], pstartsY[0])
pstartsX.pop(0)
pstartsY.pop(0)
random.shuffle(pstartsX)
random.shuffle(pstartsY)
self.canvas.move(self.id2, pstartsX[0], pstartsY[0])
pstartsX.pop(0)
pstartsY.pop(0)
random.shuffle(pstartsX)
random.shuffle(pstartsY)
self.canvas.move(self.id3, pstartsX[0], pstartsY[0])
pstartsX.pop(0)
pstartsY.pop(0)
random.shuffle(pstartsX)
random.shuffle(pstartsY)
self.y = 0
self.x = 0
self.canvas_width = self.canvas.winfo_width()
def draw(self):
self.canvas.move(self.id, self.x, 0)
pos = self.canvas.coords(self.id)
if pos[0] <= 0:
self.x = 1
elif pos[2] >= self.canvas_width:
self.x = -1
tkwidth = 500
tkheight = 400
tk.title('Level 1')
tk.resizable(0, 0)
tk.wm_attributes("-topmost", 1)
canvas = Canvas(tk, width=tkwidth, height=tkheight, bd=0, highlightthickness=0)
canvas.pack()
tk.update()
paddle = Paddle(canvas, 'blue')
ball = Ball(canvas, 3, paddle, 'green')
print(tkwidth, tkheight)
# width=tkwidth + 100, height=tkheight + 100, bd=0, highlightthickness=0
def nxt_lvl():
global pstartsX
global pstartsY
pstartsX = [100, 300, 400]
pstartsY = [125, 250, 375]
lvl2 = Toplevel(width=tkwidth + 100, height=tkheight + 100, bd=0, highlightthickness=0)
lvl2.title('Level 2')
paddle = Paddle(lvl2, 'blue')
ball = Ball(lvl2, 3, paddle, 'green')
ball.draw()
lvl_2_btn = Button(tk, text="Next Level?", command=nxt_lvl)
while 1:
if not ball.win:
ball.draw()
elif ball.win:
lvl_2_btn.pack()
tk.update_idletasks()
tk.update()
time.sleep(0.01)
考虑这段代码:
lvl2 = Toplevel(width=tkwidth + 100, height=tkheight + 100, bd=0, highlightthickness=0)
lvl2.title('Level 2')
paddle = Paddle(lvl2, 'blue')
ball = Ball(lvl2, 3, paddle, 'green')
请注意,您正在将 lvl2
传递给 Ball
和 Paddle
,而他们期望的位置是 canvas。您需要在 Toplevel
中创建一个 canvas,并将 canvas 传递给 Ball
和 Paddle
.
它应该看起来像这样:
lvl2 = Toplevel(width=tkwidth + 100, height=tkheight + 100, bd=0, highlightthickness=0)
lvl2.title('Level 2')
canvas = Canvas(lvl2)
canvas.pack(fill="both", expand=True)
paddle = Paddle(canvas, 'blue')
ball = Ball(canvas, 3, paddle, 'green')