diff --git a/VRCT.py b/VRCT.py index de84a672..8780dba8 100644 --- a/VRCT.py +++ b/VRCT.py @@ -4,13 +4,16 @@ import deepl from pythonosc import osc_message_builder from pythonosc import udp_client import customtkinter +from PIL import Image # global PATH_CONFIG = "./config.json" OSC_IP_ADDRESS = "127.0.0.1" OSC_PORT = 9000 TARGET_LANG = "EN-US" +ENABLE_TRANSLATION = True CHOICE_TRANSLATOR = "DeepL" +ENABLE_FOREGROUND = True AUTH_KEY = None TRANSLATOR = None MESSAGE_FORMAT = "[message]([translation])" @@ -25,8 +28,12 @@ if os.path.isfile(PATH_CONFIG) is not False: OSC_PORT = config["OSC_PORT"] if "TARGET_LANG" in config.keys(): TARGET_LANG = config["TARGET_LANG"] + if "ENABLE_TRANSLATION" in config.keys(): + ENABLE_TRANSLATION = config["ENABLE_TRANSLATION"] if "CHOICE_TRANSLATOR" in config.keys(): CHOICE_TRANSLATOR = config["CHOICE_TRANSLATOR"] + if "ENABLE_FOREGROUND" in config.keys(): + ENABLE_FOREGROUND = config["ENABLE_FOREGROUND"] if "AUTH_KEY" in config.keys(): AUTH_KEY = config["AUTH_KEY"] if "MESSAGE_FORMAT" in config.keys(): @@ -37,7 +44,9 @@ with open(PATH_CONFIG, 'w') as fp: "OSC_IP_ADDRESS": OSC_IP_ADDRESS, "OSC_PORT": OSC_PORT, "TARGET_LANG": TARGET_LANG, + "ENABLE_TRANSLATION": ENABLE_TRANSLATION, "CHOICE_TRANSLATOR": CHOICE_TRANSLATOR, + "ENABLE_FOREGROUND": ENABLE_FOREGROUND, "AUTH_KEY": AUTH_KEY, "MESSAGE_FORMAT": MESSAGE_FORMAT, } @@ -55,12 +64,76 @@ if AUTH_KEY is not None: customtkinter.set_appearance_mode("System") customtkinter.set_default_color_theme("blue") +class ToplevelWindow_information(customtkinter.CTkToplevel): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.grid_columnconfigure(0, weight=1) + self.grid_rowconfigure(0, weight=1) + self.geometry(f"{500}x{300}") + # self.resizable(False, False) + + self.after(200, lambda: self.iconbitmap("./img/app.ico")) + self.title("Information") + # create textbox information + self.textbox_information = customtkinter.CTkTextbox(self) + self.textbox_information.grid(row=0, column=0, padx=(10, 10), pady=(10, 10), sticky="nsew") + textbox_information_message = """VRCT(v0.2b) + +# 概要 +VRChatで使用されるChatBoxをOSC経由でメッセージを送信するツールになります。 +翻訳機能としてDeepLのAPIを使用してメッセージとその翻訳部分を同時に送信することができます。 + +# 使用方法 + 初期設定時 + 1. DeepLのAPIを使用するためにアカウント登録し、認証キーを取得する + 2. configボタンでconfigウィンドウを開きDeepL Auth Keyに認証キーを記載しcheckボタンを押す + 3. configウィンドウを閉じる + + 通常使用時 + 1. メッセージボックスにメッセージを記入 + 2. Enterキーを押し、メッセージを送信する + +# その他の設定 + コンボボックス + 翻訳機能の有効無効 + 翻訳する言語の選択 + + configウィンドウ + OSC IP address: 変更不要 + OSC port: 変更不要 + DeepL Auth key: DeepLの認証キーの設定 + Message Format: 送信するメッセージのデコレーションの設定 + [message]がメッセージボックスに記入したメッセージに置換される + [translation]が翻訳されたメッセージに置換される + 初期フォーマット:"[message]([translation])" + + 設定の初期化 + config.jsonを削除 + +# お問い合わせ +要望などはTwitterまで +https://twitter.com/misya_ai + +# アップデート履歴 +[2023-05-29: v0.1b] v0.1b リリース +[2023-05-30: v0.2b] +- 翻訳機能有効無効のチェックボックスを追加 +- 常に最前面の有効無効のチェックボックスを追加 + +# 注意事項 +再配布とかはやめてね +""" + + self.textbox_information.insert("0.0", textbox_information_message) + self.textbox_information.configure(state='disabled') + class ToplevelWindow_config(customtkinter.CTkToplevel): def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) self.geometry(f"{450}x{160}") self.resizable(False, False) + self.after(200, lambda: self.iconbitmap("./img/app.ico")) self.title("Config") self.label_ip_address = customtkinter.CTkLabel(self, text="OSC IP address:", fg_color="transparent") self.label_ip_address.grid(row=0, column=0, columnspan=1, padx=5, pady=5, sticky="nsew") @@ -136,53 +209,103 @@ class ToplevelWindow_config(customtkinter.CTkToplevel): class App(customtkinter.CTk): def __init__(self): super().__init__() - - self.title("VRC ChatBox Translator") - self.geometry(f"{400}x{180}") + self.iconbitmap('./img/app.ico') + self.title("VRCT") + self.geometry(f"{400}x{190}") self.grid_columnconfigure(1, weight=1) self.grid_rowconfigure(0, weight=1) # sidebar left self.sidebar_frame = customtkinter.CTkFrame(self, corner_radius=0) self.sidebar_frame.grid(row=0, column=0, rowspan=4, sticky="nsw") - self.sidebar_frame.grid_rowconfigure(4, weight=1) + self.sidebar_frame.grid_rowconfigure(5, weight=1) + + # checkbox translation + self.checkbox_translation = customtkinter.CTkCheckBox(self.sidebar_frame, text="translation", onvalue=True, offvalue=False, command=self.checkbox_translation_callback) + self.checkbox_translation.grid(row=0, column=0, columnspan=2 ,padx=10, pady=(5, 5), sticky="we") + + # checkbox foreground + self.checkbox_foreground = customtkinter.CTkCheckBox(self.sidebar_frame, text="foreground", onvalue=True, offvalue=False, command=self.checkbox_foreground_callback) + self.checkbox_foreground.grid(row=1, column=0, columnspan=2 ,padx=10, pady=(5, 5), sticky="we") + + # combobox translator self.combobox_translator = customtkinter.CTkComboBox(self.sidebar_frame, command=self.combobox_translator_callback) - self.combobox_translator.grid(row=0, column=0, padx=10, pady=(10, 5), sticky="w") + self.combobox_translator.grid(row=2, column=0, columnspan=2 ,padx=10, pady=(5, 5), sticky="we") + + # combobox language self.combobox_language = customtkinter.CTkComboBox(self.sidebar_frame, command=self.combobox_language_callback) - self.combobox_language.grid(row=1, column=0, padx=10, pady=(5, 10), sticky="w") - self.button_config = customtkinter.CTkButton(self.sidebar_frame, text="config", command=self.open_config) - self.button_config.grid(row=4, column=0, padx=10, pady=(10, 10), sticky="s") + self.combobox_language.grid(row=3, column=0, columnspan=2, padx=10, pady=(5, 5), sticky="we") + + # button information + self.button_information = customtkinter.CTkButton(self.sidebar_frame, text="", width=70, command=self.open_information, + image=customtkinter.CTkImage(Image.open("./img/info-icon-white.png"))) + self.button_information.grid(row=5, column=0, padx=5, pady=(10, 10), sticky="wse") + self.information_window = None + + # button config + self.button_config = customtkinter.CTkButton(self.sidebar_frame, text="", width=70, command=self.open_config, + image=customtkinter.CTkImage(Image.open("./img/config-icon-white.png"))) + self.button_config.grid(row=5, column=1, padx=5, pady=(10, 10), sticky="wse") self.config_window = None - # create textbox - self.textbox = customtkinter.CTkTextbox(self) - self.textbox.grid(row=0, column=1, padx=(10, 10), pady=(10, 5), sticky="nsew") + # create textbox message log + self.textbox_message_log = customtkinter.CTkTextbox(self) + self.textbox_message_log.grid(row=0, column=1, padx=(10, 10), pady=(10, 5), sticky="nsew") + self.textbox_message_log.configure(state='disabled') - # create entry - self.entry = customtkinter.CTkEntry(self, placeholder_text="message") - self.entry.grid(row=1, column=1, columnspan=2, padx=(10, 10), pady=(5, 10), sticky="nsew") + # create entry message box + self.entry_message_box = customtkinter.CTkEntry(self, placeholder_text="message") + self.entry_message_box.grid(row=1, column=1, columnspan=2, padx=(10, 10), pady=(5, 10), sticky="nsew") # set default values - self.combobox_translator.configure(values=["DeepL", "Disable"],) + if ENABLE_TRANSLATION: + self.checkbox_translation.select() + else: + self.checkbox_translation.deselect() + + if ENABLE_FOREGROUND: + self.checkbox_foreground.select() + self.attributes("-topmost", True) + else: + self.checkbox_foreground.deselect() + self.attributes("-topmost", False) + + self.combobox_translator.configure(values=["DeepL"],) self.combobox_language.configure(values=[ "JA","BG","CS","DA","DE","EL","EN","EN-US","EN-GB","ES","ET","FI","FR","HU", "ID","IT","KO","LT","LV","NB","NL","PL","PT","PT-BR","PT-PT","RO","RU","SK", "SL","SV","TR","UK","ZH", ],) - self.entry.bind("", self.press_key) + self.entry_message_box.bind("", self.press_key) if TRANSLATOR is None: - self.textbox.insert("0.0", f"Auth Keyを設定してないか間違っています\n") + # error update Auth key + self.textbox_message_log.configure(state='normal') + self.textbox_message_log.insert("0.0", f"Auth Keyを設定してないか間違っています\n") + self.textbox_message_log.configure(state='disabled') self.combobox_language.set(TARGET_LANG) self.combobox_translator.set(CHOICE_TRANSLATOR) - def open_config(self): if self.config_window is None or not self.config_window.winfo_exists(): self.config_window = ToplevelWindow_config(self) self.config_window.focus() + def open_information(self): + if self.information_window is None or not self.information_window.winfo_exists(): + self.information_window = ToplevelWindow_information(self) + self.information_window.focus() + + def checkbox_translation_callback(self): + global ENABLE_TRANSLATION + ENABLE_TRANSLATION = self.checkbox_translation.get() + with open(PATH_CONFIG, "r") as fp: + config = json.load(fp) + config["ENABLE_TRANSLATION"] = ENABLE_TRANSLATION + with open(PATH_CONFIG, "w") as fp: + json.dump(config, fp, indent=4) + def combobox_translator_callback(self, choice): global CHOICE_TRANSLATOR CHOICE_TRANSLATOR = choice @@ -201,18 +324,35 @@ class App(customtkinter.CTk): with open(PATH_CONFIG, "w") as fp: json.dump(config, fp, indent=4) + def checkbox_foreground_callback(self): + global ENABLE_FOREGROUND + ENABLE_FOREGROUND = self.checkbox_foreground.get() + with open(PATH_CONFIG, "r") as fp: + config = json.load(fp) + config["ENABLE_FOREGROUND"] = ENABLE_FOREGROUND + with open(PATH_CONFIG, "w") as fp: + json.dump(config, fp, indent=4) + + if ENABLE_FOREGROUND: + self.attributes("-topmost", True) + else: + self.attributes("-topmost", False) + def press_key(self, event): if TRANSLATOR is None: - self.textbox.insert("0.0", f"Auth Keyを設定してないか間違っています\n") + # error update Auth key + self.textbox_message_log.configure(state='normal') + self.textbox_message_log.insert("0.0", f"Auth Keyを設定してないか間違っています\n") + self.textbox_message_log.configure(state='disabled') else: - entry = self.entry.get() + message = self.entry_message_box.get() # translate - if CHOICE_TRANSLATOR != "Disable": - result = TRANSLATOR.translate_text(entry, target_lang=TARGET_LANG) - chat_message = MESSAGE_FORMAT.replace("[message]", entry).replace("[translation]", result.text) + if self.checkbox_translation.get() is True: + result = TRANSLATOR.translate_text(message, target_lang=TARGET_LANG) + chat_message = MESSAGE_FORMAT.replace("[message]", message).replace("[translation]", result.text) else: - chat_message = f"{entry}" + chat_message = f"{message}" # send OSC message message = osc_message_builder.OscMessageBuilder(address="/chatbox/input") @@ -223,9 +363,14 @@ class App(customtkinter.CTk): client = udp_client.SimpleUDPClient(OSC_IP_ADDRESS, OSC_PORT) client.send(message) - # delete Entry message - self.textbox.insert("0.0", f"{chat_message}\n") - self.entry.delete(0, customtkinter.END) + # update textbox message log + self.textbox_message_log.configure(state='normal') + self.textbox_message_log.insert("0.0", f"{chat_message}\n") + self.textbox_message_log.configure(state='disabled') -app = App() -app.mainloop() \ No newline at end of file + # delete message in entry message box + self.entry_message_box.delete(0, customtkinter.END) + +if __name__ == "__main__": + app = App() + app.mainloop() \ No newline at end of file diff --git a/img/config-icon-white.png b/img/config-icon-white.png new file mode 100644 index 00000000..1222f167 Binary files /dev/null and b/img/config-icon-white.png differ diff --git a/img/info-icon-white.png b/img/info-icon-white.png new file mode 100644 index 00000000..210613e0 Binary files /dev/null and b/img/info-icon-white.png differ