#!/usr/bin/env python3 # 使用系统默认的 python3 运行 ########################################################################################### # 作者:gfdgd xi # 版本:1.5.1 # 更新时间:2021年10月06日 # 感谢: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=BuildApkDeb, args=(combobox1.get(),)).start() def RunCommandShow(command): TextboxAddText1("$> {}".format(command)) TextboxAddText1(GetCommandReturn(command)) 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) apkPackageVersion = GetApkVersion(apkPath) apkChineseLabel = GetApkChineseLabel(apkPath) apkActivityName = GetApkActivityName(apkPath) iconSavePath = "{}/usr/share/uengine/icons/{}.png".format(tempPath, apkPackageName) 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(apkPackageName, apkPackageVersion, apkChineseLabel, apkChineseLabel) debPostinst = '''#!/bin/sh APK_DIR="/usr/share/uengine/apk" APK_NAME="{}.apk" APK_PATH="$APK_DIR/$APK_NAME" DESKTOP_FILE="/usr/share/applications/{}.desktop" ICON_FILE="/usr/share/uengine/icons/{}.png" if [ -f $APK_PATH ]; then echo "Installing $APK_NAME" else echo "ERROR: $APK_NAME file not found." exit 1 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 not start)." #sudo rm -f $DESKTOP_FILE #sudo rm -f $ICON_FILE #sudo rm -f "$APK_PATH" exit 1 fi ret=`/usr/bin/uengine-session-launch-helper -- uengine install --apk="$APK_PATH"` if [ $? -ne 0 ]; then echo "ERROR: apk install error..." #sudo rm -f $DESKTOP_FILE #sudo rm -f $ICON_FILE #sudo rm -f "$APK_PATH" exit 1 fi chkfail=`echo $ret |grep "Failed"` if test -n "$chkfail" ; then echo "ERROR: $ret" #sudo rm -f $DESKTOP_FILE #sudo rm -f $ICON_FILE #sudo rm -f "$APK_PATH" exit 1 fi sudo rm -f "$APK_PATH" exit 0'''.format(apkPackageName, apkPackageName, apkPackageName) debPrerm = '''#!/bin/sh APP_NAME="{}" session_manager=`ps -ef | grep "uengine session-manager" | grep -v grep` if test -z "$session_manager"; then echo "ERROR: app install failed(session-manager not start)." exit 1 fi echo "Uninstalling $APP_NAME" ret=`/usr/bin/uengine-session-launch-helper -- uengine uninstall --pkg="$APP_NAME"` if [ $? -ne 0 ]; then echo "ERROR: app uninstall error..." exit 1 fi chkfail=`echo $ret |grep "Failed"` if test -n "$chkfail" ; then echo "ERROR: $ret" exit 1 fi cat /etc/passwd | awk -F: '$3>=1000' | cut -f 1 -d : | while read line do inifile="/home/$line/.config/uengineAppGeometry.ini" if [ -f $inifile ]; then sed -i "/$APP_NAME/d" $inifile fi done exit 0'''.format(apkPackageName) 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(apkPackageName)) #write_txt("{}/usr/share/applications/{}.desktop".format(tempPath, apkPackageName), desktopFile) BuildUengineDesktop(apkPackageName, apkActivityName, apkChineseLabel, iconSavePath, "{}/usr/share/applications/{}.desktop".format(tempPath, apkPackageName)) RunCommandShow("echo '复制文件'") RunCommandShow("echo '写入 APK 软件图标'") SaveApkIcon(apkPath, iconSavePath) RunCommandShow("echo '复制 APK 文件'") RunCommandShow("cp -rv '{}' '{}/usr/share/uengine/apk/{}.apk'".format(apkPath, tempPath, apkPackageName)) 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(),apkPackageName, apkPackageVersion)) 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): # 提示:此函数有被为此程序适配而调整,如果需要最原始(无调整的)请使用主程序(此为附属组件)里的函数 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 qianZhui.get() == True: 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): things = ''' [Desktop Entry] Categories=app; Encoding=UTF-8 Exec=/usr/bin/uengine launch.sh --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 GetApkIconInApk(apkFilePath): # info = GetApkInformation(apkFilePath) # for line in info.split('\n'): # if "application:" in line: # line = line[line.index("icon='"): -1] # line = line.replace("icon='", "") # if "'" in line: # line = line[0: line.index("'")] # return line # 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) 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 = "{}/icon.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()