mirror of
https://github.com/TriM-Organization/Musicreater.git
synced 2024-11-14 11:27:42 +08:00
320 lines
10 KiB
Python
320 lines
10 KiB
Python
# -*- coding: utf-8 -*-
|
||
'''音·创的GUI窗口界面显示库
|
||
:若要使用其他界面显示,请详见:
|
||
:开发说明|指南'''
|
||
|
||
|
||
import tkinter as tk
|
||
import tkinter.simpledialog as sdialog
|
||
import tkinter.filedialog as fdialog
|
||
from msctLib.log import log
|
||
|
||
|
||
DEFAULTBLUE = (0, 137, 242)
|
||
WEAKBLUE = (0, 161, 231)
|
||
LIGHTBLUE = (38, 226, 255)
|
||
RED = (255, 52, 50)
|
||
PURPLE = (171, 112, 255)
|
||
GREEN = (0, 255, 33)
|
||
WHITE = (242, 244, 246)
|
||
BLACK = (18, 17, 16)
|
||
|
||
|
||
backgroundColor = WHITE
|
||
frontgroundColor = BLACK
|
||
loadingColor = DEFAULTBLUE
|
||
errorColor = RED
|
||
okColor = GREEN
|
||
tipsColor = PURPLE
|
||
|
||
# 注:UI界面字体、代码字体
|
||
fontPattern = ('DengXian Light', 'Fira Code')
|
||
|
||
|
||
class disp:
|
||
'''音·创 的基本Tk窗口显示库'''
|
||
|
||
def __init__(
|
||
self,
|
||
root: tk.Tk = tk.Tk(),
|
||
debug: bool = False,
|
||
title: str = '音·创',
|
||
geometry: str = '0x0',
|
||
iconbitmap: tuple = ('', ''),
|
||
menuWidget: dict = {},
|
||
wordView: str = '音·创 Musicreater',
|
||
buttons: list = [],
|
||
settingBox: list = [],
|
||
notemap: list = [],
|
||
infobar:str = '就绪',
|
||
) -> None:
|
||
'''使用参数建立基本的 音·创 窗口
|
||
:param root 根窗口
|
||
:param debug 是否将日志输出到控制台
|
||
:param title 窗口标题
|
||
wordview: str #言论部分显示的字样
|
||
button: list = [ # 操作按钮部分
|
||
dict = {
|
||
按钮名称 : tuple(按钮图标,执行函数)
|
||
},
|
||
],
|
||
settingbox: list = [ # 设置部分显示的字样及其对应的设置函数
|
||
(
|
||
设置名称:str,
|
||
值类型:tuple,
|
||
显示内容:str,
|
||
设置操作函数:<function>,
|
||
)
|
||
],
|
||
map: list = [ # 一首曲目的音符数据
|
||
音符数据
|
||
]
|
||
:param infobar 显示信息用
|
||
'''
|
||
|
||
# 载入参量 注意!图标将不被载入参数
|
||
self.root = root
|
||
'''窗口根'''
|
||
|
||
self.title = title
|
||
'''窗口标题'''
|
||
|
||
self.menuWidgets = menuWidget
|
||
'''菜单设定项'''
|
||
|
||
self.wordView = wordView
|
||
'''言·论'''
|
||
|
||
self.buttons = buttons
|
||
'''快捷功能按钮'''
|
||
|
||
self.settingBox = settingBox
|
||
'''设置框'''
|
||
|
||
self.notemap = notemap
|
||
'''音符列表'''
|
||
|
||
|
||
|
||
self.debug = debug
|
||
'''是否打开调试模式'''
|
||
|
||
self.setTitle()
|
||
self.setGeometry(geometry)
|
||
self.setIcon(*iconbitmap)
|
||
|
||
self.setMenu()
|
||
|
||
self.initWidget()
|
||
|
||
# =========================================================
|
||
# 设定函数部分
|
||
# =========================================================
|
||
|
||
def setTitle(self) -> None:
|
||
'''设置窗口标题'''
|
||
self.root.title = self.title
|
||
if self.debug:
|
||
log(f"设置窗口标题{self.title}")
|
||
|
||
def setGeometry(self,geometry:str = '0x0') -> None:
|
||
'''设置窗口大小'''
|
||
self.root.geometry(geometry)
|
||
if self.debug:
|
||
log(f"设置窗口大小{geometry}")
|
||
|
||
def setIcon(
|
||
self, bitmap: str = './musicreater.ico', default: str = ''
|
||
) -> None:
|
||
'''设置窗口图标
|
||
注意,default参数仅在Windows下有效,其意为将所有没有图标的窗口设置默认图标
|
||
如果在非Windows环境使用default参数,一个Error将被升起'''
|
||
if not self.debug:
|
||
try:
|
||
if default:
|
||
self.root.iconbitmap(bitmap, default)
|
||
log(f'设置图标为{bitmap},默认为{default}')
|
||
else:
|
||
self.root.iconbitmap(bitmap)
|
||
log(f'设置图标为{bitmap}')
|
||
return True
|
||
except Exception as e:
|
||
log(str(e), 'ERROR')
|
||
return False
|
||
else:
|
||
self.root.iconbitmap(bitmap, default)
|
||
return
|
||
|
||
def setMenu(self) -> None:
|
||
'''设置根菜单'''
|
||
if not self.menuWidgets:
|
||
# 如果传入空参数则返回当前菜单
|
||
try:
|
||
return self.RootMenu
|
||
except Exception as E:
|
||
if self.debug:
|
||
raise E
|
||
else:
|
||
log('无法读取菜单信息', 'WARRING')
|
||
# 如果不是空参数则新建菜单
|
||
self.RootMenu = {}
|
||
self.mainMenuBar = tk.Menu(self.root)
|
||
for menuName, menuCmd in self.menuWidgets.items():
|
||
# 取得一个菜单名和一堆菜单函数及其显示名称
|
||
menu = tk.Menu(self.mainMenuBar, tearoff=0)
|
||
for cmdName, cmdFunc in menuCmd.items():
|
||
if cmdName:
|
||
menu.add_command(label=cmdName, command=cmdFunc)
|
||
else:
|
||
menu.add_separator()
|
||
self.mainMenuBar.add_cascade(label=menuName, menu=menu)
|
||
self.RootMenu[menuName] = menu
|
||
self.root.config(menu=self.mainMenuBar)
|
||
|
||
def addMenu(self, menuRoot: str = '', menuLabel: str = '', menuCommand=None):
|
||
'''增加一个菜单项
|
||
:param menuRoot : str
|
||
菜单的根菜单,即所属的菜单上的文字
|
||
:param menuLabel : str
|
||
所需要增加的项目显示的文字
|
||
:param menuCommand : <function>
|
||
'''
|
||
if menuRoot in self.RootMenu.keys:
|
||
# 如果已经有父菜单
|
||
if menuLabel:
|
||
# 增加菜单指令
|
||
self.RootMenu[menuRoot].add_command(
|
||
label=menuLabel, command=menuCommand
|
||
)
|
||
else:
|
||
# 增加分隔栏
|
||
self.RootMenu[menuRoot].add_separator()
|
||
else:
|
||
# 没有父菜单则新增一个父菜单
|
||
menu = tk.Menu(self.mainMenuBar, tearoff=False)
|
||
if menuLabel:
|
||
menu.add_command(label=menuLabel, command=menuCommand)
|
||
else:
|
||
menu.add_separator()
|
||
self.mainMenuBar.add_cascade(label=menuRoot, menu=menu)
|
||
self.RootMenu[menuRoot] = menu
|
||
|
||
def initWidget(self,) -> None:
|
||
'''设置窗口小部件,分为:
|
||
:言·论 WordView
|
||
:快捷按钮面板 ButtonBar
|
||
:设置框 SettingBar
|
||
:音轨框 TrackBar
|
||
:各个音轨的显示框 TrackFrame
|
||
:信息显示版 InfoBar
|
||
'''
|
||
self._wordviewBar = tk.Label(
|
||
self.root, bg='white', fg='black', text=self.wordView, font=(fontPattern[0], 30)
|
||
)
|
||
|
||
self.setWordView(self.wordView)
|
||
|
||
def setWordView(self, text: str) -> None:
|
||
self._wordviewBar['text'] = text
|
||
|
||
|
||
# =========================================================
|
||
# 预置函数部分
|
||
# =========================================================
|
||
|
||
def authorWindowStarter(
|
||
authors: tuple = (
|
||
('金羿', 'Email EillesWan@outlook.com', 'QQ 2647547478'),
|
||
('诸葛亮与八卦阵', 'QQ 474037765'),
|
||
)
|
||
):
|
||
'''自定义作者界面'''
|
||
from languages.lang import _
|
||
from languages.lang import DEFAULTLANGUAGE
|
||
from msctLib.buildIN import version
|
||
|
||
authorWindow = tk.Tk()
|
||
authorWindow.title(_('关于'))
|
||
authorWindow.geometry('550x600') # 像素
|
||
tk.Label(authorWindow, text='', font=('', 15)).pack()
|
||
tk.Label(authorWindow, text=_('F音创'), font=('', 35)).pack()
|
||
tk.Label(
|
||
authorWindow,
|
||
text='{} {}'.format(version.version[1] + version.version[0]),
|
||
font=('', 15),
|
||
).pack()
|
||
# pack 的side可以赋值为LEFT RTGHT TOP BOTTOM
|
||
# grid 的row 是列数、column是行排,注意,这是针对空间控件本身大小来的,即是指向当前控件的第几个。
|
||
# place的 x、y是(x,y)坐标
|
||
tk.Label(
|
||
authorWindow,
|
||
image=tk.PhotoImage(file='./resources/RyounLogo.png'),
|
||
width=200,
|
||
height=200,
|
||
).pack()
|
||
tk.Label(authorWindow, text=_('凌云pairs'), font=('', 20)).pack()
|
||
tk.Label(authorWindow, text='', font=('', 15)).pack()
|
||
tk.Label(authorWindow, text=_('开发者'), font=('', 15)).pack()
|
||
for i in authors:
|
||
for j in i:
|
||
tk.Label(
|
||
authorWindow,
|
||
text=j,
|
||
font=(
|
||
'',
|
||
17 if i.index(j) == 0 else 15,
|
||
'bold' if i.index(j) == 0 else '',
|
||
),
|
||
).pack()
|
||
tk.Label(authorWindow, text='', font=('', 5)).pack()
|
||
if DEFAULTLANGUAGE != 'zh-CN':
|
||
tk.Label(authorWindow, text=_('译者'), font=('', 15)).pack()
|
||
for i in _('TRANSLATERS').split(';'):
|
||
for j in i.split(','):
|
||
tk.Label(
|
||
authorWindow,
|
||
text=j,
|
||
font=(
|
||
'',
|
||
17 if i.split(',').index(j) == 0 else 15,
|
||
'bold' if i.split(',').index(j) == 0 else '',
|
||
),
|
||
).pack()
|
||
|
||
def exitAboutWindow():
|
||
authorWindow.destroy()
|
||
|
||
tk.Button(authorWindow, text=_('确定'), command=exitAboutWindow).pack()
|
||
|
||
authorWindow.mainloop()
|
||
|
||
|
||
class ProgressBar:
|
||
def __init__(
|
||
self,
|
||
root: tk.Tk = tk.Tk(),
|
||
style: tuple = (DEFAULTBLUE, BLACK, WHITE),
|
||
type: bool = False,
|
||
info: str = '',
|
||
debug: bool = False,
|
||
) -> None:
|
||
'''建立一个进度条或者加载等待界面
|
||
:param root : tk.Tk
|
||
建立进度条的根窗口
|
||
:param style : tuple
|
||
设置主题颜色,第一个参数为进度条或者等待转圈圈的颜色,第二个参数为前景色,第三个是背景色
|
||
:param type : bool
|
||
类型,为 False 时为进度条,为 True 时为等待板
|
||
:param info : str
|
||
显示的附加信息
|
||
:param debug : bool
|
||
是否输出日志到控制台'''
|
||
self.root = root
|
||
|
||
|
||
if __name__ == '__mian__':
|
||
import os
|
||
|
||
os.chdir('../')
|
||
disp.authorMenu()
|