441 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			441 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/env python3
 | ||
| # 使用系统默认的 python3 运行
 | ||
| ###########################################################################################
 | ||
| # 作者:gfdgd xi
 | ||
| # 版本:1.5.3
 | ||
| # 更新时间:2022年02月07日(要开学了)
 | ||
| # 感谢:anbox、deepin 和 统信
 | ||
| # 基于 Python3 的 tkinter 构建
 | ||
| ###########################################################################################
 | ||
| #################
 | ||
| # 引入所需的库
 | ||
| #################
 | ||
| import os
 | ||
| import sys
 | ||
| import json
 | ||
| import shutil
 | ||
| import random
 | ||
| import zipfile
 | ||
| import traceback
 | ||
| import threading
 | ||
| import subprocess
 | ||
| import ttkthemes
 | ||
| import tkinter as tk
 | ||
| import tkinter.ttk as ttk
 | ||
| import tkinter.messagebox as messagebox
 | ||
| import tkinter.filedialog as filedialog
 | ||
| from getxmlimg import getsavexml
 | ||
| 
 | ||
| def FindApk():
 | ||
|     path = filedialog.askopenfilename(title="选择 Apk", filetypes=[("APK 文件", "*.apk"), ("所有文件", "*.*")], initialdir=json.loads(readtxt(get_home() + "/.config/uengine-runner/FindApkBuild.json"))["path"])
 | ||
|     if path != "" and path != "()":
 | ||
|         try:
 | ||
|             combobox1.set(path)
 | ||
|             write_txt(get_home() + "/.config/uengine-runner/FindApkBuild.json", json.dumps({"path": os.path.dirname(path)}))  # 写入配置文件
 | ||
|         except:
 | ||
|             pass
 | ||
| 
 | ||
| def BuildDeb():
 | ||
|     if combobox1.get() == "":
 | ||
|         messagebox.showerror(title="提示", message="信息没有填写完整,无法继续打包 APK")
 | ||
|         return
 | ||
|     if not os.path.exists(combobox1.get()):
 | ||
|         messagebox.showerror(title="提示", message="信息填写错误,无法继续打包 APK")
 | ||
|         return
 | ||
|     DisabledAndEnbled(True)
 | ||
|     threading.Thread(target=GetBuildApkDebError, args=(combobox1.get(),)).start()
 | ||
| 
 | ||
| def RunCommandShow(command):
 | ||
|     TextboxAddText1("$> {}".format(command))
 | ||
|     TextboxAddText1(GetCommandReturn(command))
 | ||
| 
 | ||
| def GetBuildApkDebError(apkPath):
 | ||
|     try:
 | ||
|         BuildApkDeb(apkPath)
 | ||
|     except:
 | ||
|         traceback.print_exc()
 | ||
|         messagebox.showerror(title="错误", message=traceback.format_exc())
 | ||
|         DisabledAndEnbled(False)
 | ||
| 
 | ||
| def BuildApkDeb(apkPath):
 | ||
|     tempPath = "/tmp/uengine-apk-builder-{}".format(int(random.randint(0, 1024)))
 | ||
|     RunCommandShow("echo '======================================New===================================='")
 | ||
|     RunCommandShow("echo '创建目录'")
 | ||
|     RunCommandShow("mkdir -pv '{}/DEBIAN'".format(tempPath))
 | ||
|     RunCommandShow("mkdir -pv '{}/usr/share/applications'".format(tempPath))
 | ||
|     RunCommandShow("mkdir -pv '{}/usr/share/uengine/apk'".format(tempPath))
 | ||
|     RunCommandShow("mkdir -pv '{}/usr/share/uengine/icons'".format(tempPath))
 | ||
|     RunCommandShow("echo '写入文件,因为写入过程过于复杂,不显示写入命令……'")
 | ||
|     apkPackageName = GetApkPackageName(apkPath, False)
 | ||
|     apkPackageNameNew = GetApkPackageName(apkPath, True)
 | ||
|     apkPackageVersion = GetApkVersion(apkPath)
 | ||
|     apkChineseLabel = GetApkChineseLabel(apkPath)
 | ||
|     apkActivityName = GetApkActivityName(apkPath)
 | ||
|     iconSavePath = "{}/usr/share/uengine/icons/{}.png".format(tempPath, apkPackageNameNew)
 | ||
|     debControl = '''Package: {}
 | ||
| Version: {}
 | ||
| Architecture: all
 | ||
| Maintainer: {}
 | ||
| Depends: deepin-elf-verify (>= 0.0.16.7-1), uengine (>= 1.0.1)
 | ||
| Section: utils
 | ||
| Priority: optional
 | ||
| Description: {}\n'''.format(apkPackageNameNew, apkPackageVersion, apkChineseLabel, apkChineseLabel)
 | ||
|     debPostinst = '''#!/bin/sh
 | ||
| 
 | ||
| APK_DIR="/usr/share/uengine/apk"
 | ||
| APK_NAME="{}"
 | ||
| APK_PATH="$APK_DIR/$APK_NAME"
 | ||
| DESKTOP_FILE="{}"
 | ||
| 
 | ||
| 
 | ||
| if [ -f $APK_PATH ]; then
 | ||
|     echo "Installing $APK_NAME"
 | ||
| else 
 | ||
|     echo "ERROR: $APK_NAME does not exist."
 | ||
|     exit 0
 | ||
| fi
 | ||
| 
 | ||
| session_manager=`ps -ef | grep "uengine session-manager" | grep -v grep`
 | ||
| if test -z "$session_manager"; then
 | ||
|     echo "ERROR: app install failed(session-manager is not running)."
 | ||
|     sess_dir="/usr/share/uengine/session_install"
 | ||
|     if [ ! -d $sess_dir ]; then
 | ||
|         mkdir $sess_dir
 | ||
|         chmod 777 $sess_dir
 | ||
|     fi
 | ||
|     apk_name=${{APK_PATH##*/}}
 | ||
|     fileName="$sess_dir/$apk_name"
 | ||
|     echo $DESKTOP_FILE > $fileName
 | ||
|     abistr=""
 | ||
|     if test -n "$abistr"; then
 | ||
|         abi=`echo $abistr |awk -F \= '{{print $2}}'`
 | ||
|         echo $abi >> $fileName
 | ||
|     fi
 | ||
|     chmod 766 $fileName
 | ||
| fi
 | ||
| 
 | ||
| /usr/bin/uengine-session-launch-helper -- uengine install  --apk="$APK_PATH"
 | ||
| 
 | ||
| exit 0'''.format(apkPackageNameNew + ".apk", "/usr/share/applications/{}.desktop".format(apkPackageNameNew))
 | ||
|     debPrerm = '''#!/bin/sh
 | ||
| 
 | ||
| APP_NAME="{}"
 | ||
| DESKTOP_FILE="{}"
 | ||
| 
 | ||
| session_manager=`ps -ef | grep "uengine session-manager" | grep -v grep`
 | ||
| if test -z "$session_manager"; then
 | ||
|     echo "ERROR: app uninstall failed(session-manager is not running)."
 | ||
|     sess_dir="/usr/share/uengine/session_uninstall"
 | ||
|     if [ ! -d $sess_dir ]; then
 | ||
|         mkdir $sess_dir
 | ||
|         chmod 777 $sess_dir
 | ||
|     fi
 | ||
|     fileName="$sess_dir/$APP_NAME"
 | ||
|     echo $DESKTOP_FILE > $fileName
 | ||
|     chmod 766 $fileName
 | ||
| fi
 | ||
| 
 | ||
| echo "Uninstalling $APP_NAME"
 | ||
| /usr/bin/uengine-session-launch-helper -- uengine uninstall --pkg="$APP_NAME"
 | ||
| 
 | ||
| exit 0'''.format(apkPackageName, "/usr/share/applications/{}.desktop".format(apkPackageNameNew))
 | ||
|     desktopFile = '''[Desktop Entry]
 | ||
| Categories=Other;
 | ||
| Exec=uengine launch --action=android.intent.action.MAIN --package={} --component={}
 | ||
| Icon=/usr/share/uengine/icons/{}.png
 | ||
| Terminal=false
 | ||
| Type=Application
 | ||
| GenericName={}
 | ||
| Name={}
 | ||
| '''
 | ||
|     #RunCommandShow("echo '{}' > '{}/DEBIAN/control'".format(debControl, tempPath))
 | ||
|     RunCommandShow("echo 正在写入文件:'{}/DEBIAN/control'".format(tempPath))
 | ||
|     write_txt("{}/DEBIAN/control".format(tempPath), debControl)
 | ||
|     RunCommandShow("echo 正在写入文件:'{}/DEBIAN/postinst'".format(tempPath))
 | ||
|     write_txt("{}/DEBIAN/postinst".format(tempPath), debPostinst)
 | ||
|     RunCommandShow("echo 正在写入文件:'{}/DEBIAN/prerm'".format(tempPath))
 | ||
|     write_txt("{}/DEBIAN/prerm".format(tempPath), debPrerm)
 | ||
|     RunCommandShow("echo 正在写入文件:'/usr/share/applications/{}.desktop'".format(apkPackageNameNew))
 | ||
|     #write_txt("{}/usr/share/applications/{}.desktop".format(tempPath, apkPackageNameNew), desktopFile)
 | ||
|     BuildUengineDesktop(apkPackageName, apkActivityName, apkChineseLabel, "/usr/share/uengine/icons/{}.png".format(apkPackageNameNew),
 | ||
|                         "{}/usr/share/applications/{}.desktop".format(tempPath, apkPackageNameNew))
 | ||
|     RunCommandShow("echo '复制文件'")
 | ||
|     RunCommandShow("echo '写入 APK 软件图标'")
 | ||
|     SaveApkIcon(apkPath, iconSavePath)
 | ||
|     RunCommandShow("echo '复制 APK 文件'")
 | ||
|     RunCommandShow("cp -rv '{}' '{}/usr/share/uengine/apk/{}.apk'".format(apkPath, tempPath, apkPackageNameNew))
 | ||
|     RunCommandShow("echo '正在设置文件权限……'")
 | ||
|     RunCommandShow("chmod 0775 -vR '{}/DEBIAN/postinst'".format(tempPath))
 | ||
|     RunCommandShow("chmod 0775 -vR '{}/DEBIAN/prerm'".format(tempPath))
 | ||
|     RunCommandShow("echo '打包 deb 到桌面……'")
 | ||
|     RunCommandShow("dpkg -b '{}' '{}/{}_{}.deb'".format(tempPath, get_desktop_path(),apkPackageNameNew, apkPackageVersion))
 | ||
|     RunCommandShow("echo '正在删除临时目录……'")
 | ||
|     RunCommandShow("rm -rfv '{}'".format(tempPath))
 | ||
|     RunCommandShow("echo '完成!'")
 | ||
|     findApkHistory.append(apkPath)
 | ||
|     combobox1['value'] = findApkHistory
 | ||
|     write_txt(get_home() + "/.config/uengine-runner/FindApkBuildHistory.json", str(json.dumps(ListToDictionary(findApkHistory))))  # 将历史记录的数组转换为字典并写入
 | ||
|     messagebox.showinfo(title="提示", message="打包完成")
 | ||
|     DisabledAndEnbled(False)
 | ||
| 
 | ||
| def DisabledAndEnbled(choose):
 | ||
|     userChoose = {True: tk.DISABLED, False: tk.NORMAL}
 | ||
|     a = userChoose[choose]
 | ||
|     combobox1.configure(state=a)
 | ||
|     check.configure(state=a)
 | ||
|     button2.configure(state=a)
 | ||
|     button3.configure(state=a)
 | ||
| 
 | ||
| # 需引入 subprocess
 | ||
| def GetCommandReturn(cmd):
 | ||
|     # cmd 是要获取输出的命令
 | ||
|     return subprocess.getoutput(cmd)
 | ||
| 
 | ||
| # 重启本应用程序
 | ||
| def ReStartProgram():
 | ||
|     python = sys.executable
 | ||
|     os.execl(python, python, * sys.argv)
 | ||
| 
 | ||
| # 获取用户主目录
 | ||
| def get_home():
 | ||
|     return os.path.expanduser('~')
 | ||
| 
 | ||
| # 获取当前语言
 | ||
| def get_now_lang()->"获取当前语言":
 | ||
|     return os.getenv('LANG')
 | ||
| 
 | ||
| # 获取用户桌面目录
 | ||
| def get_desktop_path():
 | ||
|     for line in open(get_home() + "/.config/user-dirs.dirs"):  # 以行来读取配置文件
 | ||
|         desktop_index = line.find("XDG_DESKTOP_DIR=\"")  # 寻找是否有对应项,有返回 0,没有返回 -1
 | ||
|         if desktop_index != -1:  # 如果有对应项
 | ||
|             break  # 结束循环
 | ||
|     if desktop_index == -1:  # 如果是提前结束,值一定≠-1,如果是没有提前结束,值一定=-1
 | ||
|         return -1
 | ||
|     else:
 | ||
|         get = line[17:-2]  # 截取桌面目录路径
 | ||
|         get_index = get.find("$HOME")  # 寻找是否有对应的项,需要替换内容
 | ||
|         if get != -1:  # 如果有
 | ||
|             get = get.replace("$HOME", get_home())  # 则把其替换为用户目录(~)
 | ||
|         return get  # 返回目录
 | ||
| 
 | ||
| # 数组转字典
 | ||
| def ListToDictionary(list):
 | ||
|     dictionary = {}
 | ||
|     for i in range(len(list)):
 | ||
|         dictionary[i] = list[i]
 | ||
|     return dictionary
 | ||
| 
 | ||
| # 读取文本文档
 | ||
| def readtxt(path):
 | ||
|     f = open(path, "r")  # 设置文件对象
 | ||
|     str = f.read()  # 获取内容
 | ||
|     f.close()  # 关闭文本对象
 | ||
|     return str  # 返回结果
 | ||
| 
 | ||
| # 写入文本文档
 | ||
| def write_txt(path, things):
 | ||
|     file = open(path, 'w', encoding='UTF-8')  # 设置文件对象
 | ||
|     file.write(things)  # 写入文本
 | ||
|     file.close()  # 关闭文本对象
 | ||
| 
 | ||
| def GetApkInformation(apkFilePath):
 | ||
|     return GetCommandReturn("aapt dump badging '{}'".format(apkFilePath))
 | ||
| 
 | ||
| def GetApkActivityName(apkFilePath):
 | ||
|     info = GetApkInformation(apkFilePath)
 | ||
|     for line in info.split('\n'):
 | ||
|         if "launchable-activity" in line:
 | ||
|             line = line[0: line.index("label='")]
 | ||
|             line = line.replace("launchable-activity: ", "")
 | ||
|             line = line.replace("'", "")
 | ||
|             line = line.replace(" ", "")
 | ||
|             line = line.replace("name=", "")
 | ||
|             line = line.replace("label=", "")
 | ||
|             line = line.replace("icon=", "")
 | ||
|             return line
 | ||
| 
 | ||
| def GetApkPackageName(apkFilePath, setting):
 | ||
|     # 提示:此函数有被为此程序适配而调整,如果需要最原始(无调整的)请使用主程序(此为附属组件)里的函数
 | ||
|     info = GetApkInformation(apkFilePath)
 | ||
|     for line in info.split('\n'):
 | ||
|         if "package:" in line:
 | ||
|             line = line[0: line.index("versionCode='")]
 | ||
|             line = line.replace("package:", "")
 | ||
|             line = line.replace("name=", "")
 | ||
|             line = line.replace("'", "")
 | ||
|             line = line.replace(" ", "")
 | ||
|             # 此较为特殊,因为需要判断用户是否要添加前缀
 | ||
|             if setting:
 | ||
|                 return "uengine-dc-{}".format(line).lower()
 | ||
|             return line.lower()
 | ||
| 
 | ||
| def GetApkVersion(apkFilePath):
 | ||
|     info = GetApkInformation(apkFilePath)
 | ||
|     for line in info.split('\n'):
 | ||
|         if "package:" in line:
 | ||
|             if "compileSdkVersion='" in line:
 | ||
|                 line = line.replace(line[line.index("compileSdkVersion='"): -1], "")
 | ||
|             if "platform" in line:
 | ||
|                 line = line.replace(line[line.index("platform"): -1], "")
 | ||
|             line = line.replace(line[0: line.index("versionName='")], "")
 | ||
|             line = line.replace("versionName='", "")
 | ||
|             line = line.replace("'", "")
 | ||
|             line = line.replace(" ", "")
 | ||
|             return line
 | ||
| 
 | ||
| def BuildUengineDesktop(packageName, activityName, showName, iconPath, savePath):
 | ||
|     if showName == "" or showName == None:
 | ||
|         showName = "未知应用"
 | ||
|     things = '''
 | ||
|     [Desktop Entry]
 | ||
| Categories=app;
 | ||
| Encoding=UTF-8
 | ||
| Exec=/usr/bin/uengine launch --action=android.intent.action.MAIN --package={} --component={}
 | ||
| GenericName={}
 | ||
| Icon={}
 | ||
| MimeType=
 | ||
| Name={}
 | ||
| StartupWMClass={}
 | ||
| Terminal=false
 | ||
| Type=Application
 | ||
| '''.format(packageName, activityName, showName, iconPath, showName, showName)
 | ||
|     write_txt(savePath, things)
 | ||
| 
 | ||
| def GetApkChineseLabel(apkFilePath):
 | ||
|     info = GetApkInformation(apkFilePath)
 | ||
|     for line in info.split('\n'):
 | ||
|         if "application-label:" in line:
 | ||
|             line = line.replace("application-label:", "")
 | ||
|             line = line.replace("'", "")
 | ||
|             return line
 | ||
| 
 | ||
| #合并两个函数到一起
 | ||
| def SaveApkIcon(apkFilePath, iconSavePath)->"获取 apk 文件的图标":
 | ||
|     try:
 | ||
|         info = GetApkInformation(apkFilePath)
 | ||
|         for line in info.split('\n'):
 | ||
|             if "application:" in line:
 | ||
|                 xmlpath = line.split(":")[-1].split()[-1].split("=")[-1].replace("'","")  
 | ||
|                 if xmlpath.endswith('.xml'):
 | ||
|                         xmlsave = getsavexml()
 | ||
|                         print(xmlpath)
 | ||
|                         xmlsave.savexml(apkFilePath,xmlpath,iconSavePath)
 | ||
|                 else:
 | ||
|                     zip = zipfile.ZipFile(apkFilePath)
 | ||
|                     iconData = zip.read(xmlpath)
 | ||
|                     with open(iconSavePath, 'w+b') as saveIconFile:
 | ||
|                         saveIconFile.write(iconData)
 | ||
|                 return
 | ||
|         print("Show defult icon")
 | ||
|         shutil.copy(programPath + "/defult.png", iconSavePath)
 | ||
|     except:
 | ||
|         traceback.print_exc()
 | ||
|         print("Error, show defult icon")
 | ||
|         shutil.copy(programPath + "/defult.png", iconSavePath)
 | ||
| 
 | ||
| 
 | ||
| #def SaveApkIcon(apkFilePath, iconSavePath):
 | ||
| #    zip = zipfile.ZipFile(apkFilePath)
 | ||
| #    iconData = zip.read(GetApkIconInApk(apkFilePath))
 | ||
| #    with open(iconSavePath, 'w+b') as saveIconFile:
 | ||
| #        saveIconFile.write(iconData)
 | ||
| 
 | ||
| def TextboxAddText1(message):
 | ||
|     global textbox1
 | ||
|     textbox1.configure(state=tk.NORMAL)
 | ||
|     textbox1.insert(tk.END,message + "\n")
 | ||
|     textbox1.configure(state=tk.DISABLED)    
 | ||
| 
 | ||
| # 获取用户桌面目录
 | ||
| def get_desktop_path():
 | ||
|     for line in open(get_home() + "/.config/user-dirs.dirs"):  # 以行来读取配置文件
 | ||
|         desktop_index = line.find("XDG_DESKTOP_DIR=\"")  # 寻找是否有对应项,有返回 0,没有返回 -1
 | ||
|         if desktop_index != -1:  # 如果有对应项
 | ||
|             break  # 结束循环
 | ||
|     if desktop_index == -1:  # 如果是提前结束,值一定≠-1,如果是没有提前结束,值一定=-1
 | ||
|         return -1
 | ||
|     else:
 | ||
|         get = line[17:-2]  # 截取桌面目录路径
 | ||
|         get_index = get.find("$HOME")  # 寻找是否有对应的项,需要替换内容
 | ||
|         if get != -1:  # 如果有
 | ||
|             get = get.replace("$HOME", get_home())  # 则把其替换为用户目录(~)
 | ||
|         return get  # 返回目录
 | ||
| 
 | ||
| # 获取用户主目录
 | ||
| def get_home():
 | ||
|     return os.path.expanduser('~')
 | ||
| 
 | ||
| ###########################
 | ||
| # 程序信息
 | ||
| ###########################
 | ||
| programPath = os.path.split(os.path.realpath(__file__))[0]  # 返回 string
 | ||
| lang = get_now_lang()
 | ||
| langFile = json.loads(readtxt(programPath + "/Language.json"))
 | ||
| if not lang in langFile.keys():
 | ||
|     lang = "en_US.UTF-8"
 | ||
| information = json.loads(readtxt(programPath + "/information.json"))
 | ||
| version = information["Version"]
 | ||
| title = "{} {}".format(langFile[lang]["Uengine Apk Builder"]["Title"], version)
 | ||
| iconPath = "{}/builer.png".format(os.path.split(os.path.realpath(__file__))[0])
 | ||
| 
 | ||
| ###########################
 | ||
| # 加载配置
 | ||
| ###########################
 | ||
| if not os.path.exists(get_home() + "/.config/uengine-runner"):  # 如果没有配置文件夹
 | ||
|     os.makedirs(get_home() + "/.config/uengine-runner")  # 创建配置文件夹
 | ||
| if not os.path.exists(get_home() + "/.config/uengine-runner/FindApkBuildHistory.json"):  # 如果没有配置文件
 | ||
|     write_txt(get_home() + "/.config/uengine-runner/FindApkBuildHistory.json", json.dumps({}))  # 创建配置文件
 | ||
| if not os.path.exists(get_home() + "/.config/uengine-runner/FindApkBuild.json"):  # 如果没有配置文件
 | ||
|     write_txt(get_home() + "/.config/uengine-runner/FindApkBuild.json", json.dumps({"path": "~"}))  # 创建配置文件
 | ||
| 
 | ||
| ###########################
 | ||
| # 设置变量
 | ||
| ###########################
 | ||
| findApkHistory = list(json.loads(readtxt(get_home() + "/.config/uengine-runner/FindApkBuildHistory.json")).values())
 | ||
| 
 | ||
| ###########################
 | ||
| # 窗口创建
 | ||
| ###########################
 | ||
| win = tk.Tk()
 | ||
| 
 | ||
| qianZhui = tk.BooleanVar()
 | ||
| 
 | ||
| window = ttk.Frame(win)
 | ||
| frame2 = ttk.Frame(window)
 | ||
| label1 = ttk.Label(window, text=langFile[lang]["Uengine Apk Builder"]["label1"])
 | ||
| combobox1 = ttk.Combobox(window, width=100)
 | ||
| button2 = ttk.Button(window, text=langFile[lang]["Uengine Apk Builder"]["button2"], command=FindApk)
 | ||
| button3 = ttk.Button(frame2, text=langFile[lang]["Uengine Apk Builder"]["button3"], command=BuildDeb)
 | ||
| check = ttk.Checkbutton(frame2, variable=qianZhui,text=langFile[lang]["Uengine Apk Builder"]["check"])
 | ||
| textbox1 = tk.Text(window, width=100)
 | ||
| menu = tk.Menu(window, background="white")  # 设置菜单栏
 | ||
| programmenu = tk.Menu(menu, tearoff=0, background="white")  # 设置“程序”菜单栏
 | ||
| 
 | ||
| menu.add_cascade(label=langFile[lang]["Uengine Apk Builder"]["Menu"][0]["Name"], menu=programmenu)
 | ||
| programmenu.add_command(label=langFile[lang]["Uengine Apk Builder"]["Menu"][0]["Menu"][0], command=window.quit)  # 设置“退出程序”项
 | ||
| # 设置控件
 | ||
| combobox1['value'] = findApkHistory
 | ||
| textbox1.configure(state=tk.DISABLED)
 | ||
| textbox1.config(foreground='white', background='black')
 | ||
| # 如果有参数
 | ||
| if len(sys.argv) > 1:
 | ||
|     combobox1.set(sys.argv[1])
 | ||
| # 设置窗口
 | ||
| style = ttkthemes.ThemedStyle(win)
 | ||
| style.set_theme("breeze")
 | ||
| #win.attributes('-alpha', 0.5)
 | ||
| win.title(title)
 | ||
| win.resizable(0, 0)
 | ||
| win.iconphoto(False, tk.PhotoImage(file=iconPath))
 | ||
| #
 | ||
| win.config(menu=menu)  # 显示菜单栏
 | ||
| label1.grid(row=2, column=0)
 | ||
| combobox1.grid(row=2, column=1)
 | ||
| button2.grid(row=2, column=2)
 | ||
| button3.grid(row=0, column=1)
 | ||
| check.grid(row=0, column=0)
 | ||
| frame2.grid(row=3, columnspa=3)
 | ||
| textbox1.grid(row=4, columnspa=3)
 | ||
| window.pack()
 | ||
| win.mainloop() |