привет, я работаю над небольшим проектом и сделал небольшой экранный кексборд под именем tkinter Toplevel
мое приложение выглядит так:
- Root - Winow (Tk-Widget)
- ввод 1 (Entry-Widget)
- ввод 2 (Entry-Widget)
- ввод 3 (Text-Widget)
- on_screen-клавиатура (Toplevel-Widget)
Toplevel-Widget содержит кнопки, обратные вызовы которых должны вводить текст в записи (так же, как кнопки клавиатуры)
То, что я хочу, это связь между дочерними элементами клавиатуры / клавиатуры и последним активным виджетом ввода . Моя проблема в том, что я не знаю, как сказать клавиатуру, для которого Widget ввода должен послать сообщение.
import tkinter as tk
from tkinter import ttk
class MainWindow(tk.Tk):
def __init__(self, *args, **kwargs):
tk.Tk.__init__(self, *args, **kwargs)
self.active_input = tk.Variable(value=None)
ttk.Button(self, text="show Keyboard", command=lambda: Keyboard(self)).pack()
self.text = tk.StringVar(value="")
self.input1 = ttk.Entry(self)
self.input1.bind("<FocusIn>", lambda e: self.active_input.set(self.input1))
self.input2 = ttk.Entry(self)
self.input2.bind("<FocusIn>", lambda e: self.active_input.set(self.input2))
self.input3 = tk.Text(self, height=3, width=15)
self.input3.bind("<FocusIn>", lambda e: self.active_input.set(self.input3))
self.input1.pack()
self.input3.pack()
self.input2.pack()
class Keyboard(tk.Toplevel):
OPENED = False
NAME = "- Keyboard -"
NUM = [{"text":"1", "width":1},
{"text":"2", "width":1},
{"text":"3", "width":2}]
CHAR= [{"text":"A", "width":1},
{"text":"B", "width":1},
{"text":"C", "width":2}]
def __init__(self, master):
if not Keyboard.OPENED:
Keyboard.OPENED = True
print("keyboard opened!")
self.master = master
tk.Toplevel.__init__(self, master)
self.title(self.NAME)
self.protocol("WM_DELETE_WINDOW", self.close)
self.keyb_nb = ttk.Notebook(self)
self.keyb_nb.pack()
self.num_tab = ttk.Frame(self.keyb_nb)
self.createPad(self.num_tab, Keyboard.NUM,2)
self.keyb_nb.add(self.num_tab, text="123")
self.char_tab = ttk.Frame(self.keyb_nb)
self.createPad(self.char_tab, Keyboard.CHAR, 2)
self.keyb_nb.add(self.char_tab, text="ABC")
def createPad(self, master, pad:list, max_col):
self.co_count = 0
self.ro = 1
for button in pad:
button["id"] = ttk.Button(master, width=6*button["width"], text=button["text"], command=self.bclicked(button))
if self.co_count >= max_col:
self.ro = self.ro + 1
self.co_count = 0
button["id"].grid(row=self.ro, columnspan=button["width"], column=self.co_count)
self.co_count = self.co_count+button["width"]
def bclicked(self, button:dict):
"""
reciver = self.master.active_input #I think the Problem here is, that the variable contains a string, not a widget
reciever.focus_force()
reciever.insert(index=tk.INSERT, string=button["text"])
"""
pass
def close(self):
Keyboard.OPENED = False
self.destroy()
print("keyboard closed!")
root = MainWindow()
root.mainloop()
Здесь инициализация Mainwindow и bclicked класса Keyboard важны ...
код готов к отладке
я бы предпочел решение, похожее на сообщение в inte rnet (отправитель = кнопка, идентификатор получателя, сообщение) , но очень приветствую каждое рабочее решение
кстати: я также ищу решение, как мне не нужно заставлять ввод фокусироваться, а Toplevel остается на самом высоком уровне экрана (что если я сфокусируйте Tk-Widget / один из входов / кнопку, клавиатура останется перед ней)
спасибо за вашу помощь;)
РЕЗЮМЕ: как мне узнать, что из 3 виджета ввода были активны, наконец, когда уровень клавиатуры уже имеет фокус?