410 lines
20 KiB
Python
Raw Normal View History

2021-01-29 12:07:03 +08:00
#!/usr/bin/env python3
# 使用系统默认的 python3 运行
###########################################################################################
# 作者gfdgd xi
2022-07-03 18:38:02 +08:00
# 版本1.5.0
# 更新时间2022年07月03日
# 感谢:感谢 wine 以及 deepin-wine 团队,提供了 wine 和 deepin-wine 给大家使用,让我能做这个程序
# 基于 Python3 的 tkinter 构建
###########################################################################################
2021-01-29 12:07:03 +08:00
#################
# 引入所需的库
#################
import os
2021-05-23 15:54:04 +08:00
import sys
import time
import json
import shutil
import threading
2021-07-27 15:05:19 +08:00
import ttkthemes
2021-03-13 09:55:49 +08:00
import webbrowser
2022-07-03 18:38:02 +08:00
import subprocess
2021-07-27 15:05:19 +08:00
import tkinter as tk
import tkinter.ttk as ttk
import tkinter.filedialog
import tkinter.messagebox
import PIL.Image as Image
import PIL.ImageTk as ImageTk
2021-01-29 12:07:03 +08:00
###################
# 程序所需事件
###################
2021-03-13 09:55:49 +08:00
2021-05-23 15:54:04 +08:00
# 打开程序官网
2021-03-13 09:55:49 +08:00
def OpenProgramURL():
2021-05-23 15:54:04 +08:00
webbrowser.open_new_tab(programUrl)
2021-03-13 09:55:49 +08:00
# 读取文本文档
def readtxt(path):
f = open(path, "r") # 设置文件对象
str = f.read() # 获取内容
f.close() # 关闭文本对象
return str # 返回结果
# 写入文本文档
def write_txt(path, things):
2021-05-23 15:54:04 +08:00
file = open(path, 'w', encoding='UTF-8') # 设置文件对象
file.write(things) # 写入文本
file.close() # 关闭文本对象
# 获取用户桌面目录
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('~')
# 第一个浏览按钮事件
2021-01-29 12:07:03 +08:00
def liulanbutton():
2021-05-23 15:54:04 +08:00
path = tkinter.filedialog.askdirectory(title="选择 wine 容器", initialdir=json.loads(readtxt(get_home() + "/.config/deepin-wine-runner/WineBotton.json"))["path"])
if path != "" and path != "()":
e1.set(path)
write_txt(get_home() + "/.config/deepin-wine-runner/WineBotton.json", json.dumps({"path": path})) # 写入配置文件
2021-01-29 12:07:03 +08:00
# 第二个浏览按钮事件
2021-01-29 12:07:03 +08:00
def liulanexebutton():
2021-05-23 15:54:04 +08:00
path = tkinter.filedialog.askopenfilename(title="选择 exe 可执行文件", filetypes=[("exe 可执行文件", "*.exe"), ("EXE 可执行文件", "*.EXE"), ("所有文件", "*.*")], initialdir=json.loads(readtxt(get_home() + "/.config/deepin-wine-runner/FindExe.json"))["path"])
if path != "" and path != "()":
e2.set(path) # 显示路径
write_txt(get_home() + "/.config/deepin-wine-runner/FindExe.json", json.dumps({"path": os.path.dirname(path)})) # 写入配置文件
2021-01-29 12:07:03 +08:00
2021-05-23 15:54:04 +08:00
# 使用多线程运行可执行文件
2021-01-29 12:07:03 +08:00
def runexebutton():
run = threading.Thread(target=runexebutton_threading)
run.start()
2021-03-13 09:55:49 +08:00
def DisableButton(things):
a = {True: tk.DISABLED, False: tk.NORMAL}
button1.configure(state=a[things])
button2.configure(state=a[things])
button3.configure(state=a[things])
e1.configure(state=a[things])
e2.configure(state=a[things])
o1.configure(state=a[things])
2022-07-03 18:38:02 +08:00
uninstallProgram.configure(state=a[things])
2021-03-13 09:55:49 +08:00
# 运行可执行文件的线程
def runexebutton_threading():
2021-03-13 09:55:49 +08:00
DisableButton(True)
2021-05-23 15:54:04 +08:00
if e1.get() == "" or e2.get() == "": # 判断文本框是否有内容
tkinter.messagebox.showinfo(title="提示", message="没有填写需要使用的 wine 容器或需要运行的 exe 应用")
2022-07-03 18:38:02 +08:00
DisableButton(False)
return
else: # 如果都有
2022-07-03 18:38:02 +08:00
#text = subprocess.getoutput("WINEPREFIX='" + e1.get() + "' " + wine[o1_text.get()] + " '" + e2.get() + "'") # 运行
res = subprocess.Popen(["WINEPREFIX='" + e1.get() + "' " + wine[o1_text.get()] + " '" + e2.get() + "'"], shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
# 清空文本框内容
returnText.config(state=tk.NORMAL)
returnText.delete(1.0, "end")
returnText.config(state=tk.DISABLED)
# 实时读取程序返回
while res.poll() is None:
returnText.config(state=tk.NORMAL)
text = res.stdout.readline().decode("utf8")
returnText.insert("end", text)
print(text)
returnText.config(state=tk.DISABLED)
2021-05-23 15:54:04 +08:00
findExeHistory.append(e1.get()) # 将记录写进数组
wineBottonHistory.append(e2.get()) # 将记录写进数组
write_txt(get_home() + "/.config/deepin-wine-runner/FindExeHistory.json", str(json.dumps(ListToDictionary(findExeHistory)))) # 将历史记录的数组转换为字典并写入
write_txt(get_home() + "/.config/deepin-wine-runner/WineBottonHistory.json", str(json.dumps(ListToDictionary(wineBottonHistory)))) # 将历史记录的数组转换为字典并写入
e1['value'] = findExeHistory
e2['value'] = wineBottonHistory
2021-03-13 09:55:49 +08:00
DisableButton(False)
2021-01-29 12:07:03 +08:00
# 显示“关于这个程序”窗口
2021-07-27 15:05:19 +08:00
def about_this_program()->"显示“关于这个程序”窗口":
2021-05-23 15:54:04 +08:00
global about
2021-07-27 15:05:19 +08:00
global title
global iconPath
mess = tk.Toplevel()
message = ttk.Frame(mess)
mess.resizable(0, 0)
mess.title("关于 {}".format(title))
mess.iconphoto(False, tk.PhotoImage(file=iconPath))
img = ImageTk.PhotoImage(Image.open(iconPath))
label1 = ttk.Label(message, image=img)
label2 = ttk.Label(message, text=about)
button1 = ttk.Button(message, text="确定", command=mess.withdraw)
label1.pack()
label2.pack()
button1.pack(side="bottom")
message.pack()
mess.mainloop()
2021-01-29 12:07:03 +08:00
# 显示“提示”窗口
def helps():
2021-05-23 15:54:04 +08:00
global tips
tkinter.messagebox.showinfo(title="提示", message=tips)
# 显示更新内容窗口
def UpdateThings():
tkinter.messagebox.showinfo(title="更新内容", message=updateThings)
2022-07-03 18:38:02 +08:00
# 生成 desktop 文件在启动器
def make_desktop_on_launcher():
if combobox1.get() == "" or e2.get() == "" or e1.get() == "": # 判断文本框是否有内容
tkinter.messagebox.showinfo(title="提示", message="没有填写需要使用的 wine 容器或需要运行的 exe 应用或保存的文件名")
else: # 如果都有
if os.path.exists(get_home() + "/.local/share/applications/" + combobox1.get() + ".desktop"): # 判断目录是否有该文件,如果有
choose = tkinter.messagebox.askokcancel(title="提示", message="文件已经存在,是否覆盖?") # 询问用户是否覆盖
if choose: # 如要覆盖
os.remove(get_home() + "/.local/share/applications/" + combobox1.get() + ".desktop") # 删除该文件
else: # 如不覆盖
return # 结束
#os.mknod(get_home() + "/.local/share/applications/" + combobox1.get() + ".desktop")
write_txt(get_home() + "/.local/share/applications/" + combobox1.get() + ".desktop", f'''[Desktop Entry]
Name={combobox1.get()}
Exec=env WINEPREFIX='{e1.get()}' {wine[o1_text.get()]} '{e2.get()}'
Icon={iconPath}
Type=Application
StartupNotify=true''') # 写入文本文档
shellHistory.append(combobox1.get()) # 将记录写进数组
write_txt(get_home() + "/.config/deepin-wine-runner/ShellHistory.json", str(json.dumps(ListToDictionary(shellHistory)))) # 将历史记录的数组转换为字典并写入
combobox1['value'] = shellHistory
tkinter.messagebox.showinfo(title="提示", message="生成完成!") # 显示完成对话框
# 生成 desktop 文件在桌面
# (第四个按钮的事件)
def make_desktop_on_desktop():
2021-05-23 15:54:04 +08:00
if combobox1.get() == "" or e2.get() == "" or e1.get() == "": # 判断文本框是否有内容
tkinter.messagebox.showinfo(title="提示", message="没有填写需要使用的 wine 容器或需要运行的 exe 应用或保存的文件名")
else: # 如果都有
2022-07-03 18:38:02 +08:00
if os.path.exists(get_desktop_path() + "/" + combobox1.get() + ".desktop"): # 判断目录是否有该文件,如果有
2021-03-13 09:55:49 +08:00
choose = tkinter.messagebox.askokcancel(title="提示", message="文件已经存在,是否覆盖?") # 询问用户是否覆盖
if choose: # 如要覆盖
2022-07-03 18:38:02 +08:00
os.remove(get_desktop_path() + "/" + combobox1.get() + ".desktop") # 删除该文件
else: # 如不覆盖
return # 结束
2022-07-03 18:38:02 +08:00
os.mknod(get_desktop_path() + "/" + combobox1.get() + ".desktop")
write_txt(get_desktop_path() + "/" + combobox1.get() + ".desktop", f'''[Desktop Entry]
Name={combobox1.get()}
Exec=env WINEPREFIX='{e1.get()}' {wine[o1_text.get()]} '{e2.get()}'
Icon={iconPath}
Type=Application
StartupNotify=true''') # 写入文本文档
2021-05-23 15:54:04 +08:00
shellHistory.append(combobox1.get()) # 将记录写进数组
write_txt(get_home() + "/.config/deepin-wine-runner/ShellHistory.json", str(json.dumps(ListToDictionary(shellHistory)))) # 将历史记录的数组转换为字典并写入
combobox1['value'] = shellHistory
2021-03-13 09:55:49 +08:00
tkinter.messagebox.showinfo(title="提示", message="生成完成!") # 显示完成对话框
2021-01-29 12:07:03 +08:00
2021-05-23 15:54:04 +08:00
# 数组转字典
def ListToDictionary(list):
dictionary = {}
for i in range(len(list)):
dictionary[i] = list[i]
return dictionary
def CleanProgramHistory():
if tkinter.messagebox.askokcancel(title="警告", message="删除后将无法恢复,你确定吗?\n删除后软件将会自动重启。"):
shutil.rmtree(get_home() + "/.config/deepin-wine-runner")
ReStartProgram()
# 重启本应用程序
def ReStartProgram():
python = sys.executable
os.execl(python, python, * sys.argv)
2022-07-03 18:38:02 +08:00
def KillProgram():
os.system(f"killall {wine[o1_text.get()]} -9")
os.system("killall winedbg -9")
def InstallWine():
threading.Thread(target=os.system, args=[f"deepin-terminal -e \"{programPath}/AllInstall.py\""]).start()
def OpenWineBotton():
if e1.get() == "":
tkinter.messagebox.showinfo(title="提示", message="您未选择需要打开的 Wine 容器")
return
os.system("xdg-open \"" + e1.get().replace("\"", "\\\"") + "\"")
def OpenWineFontPath():
if e1.get() == "":
tkinter.messagebox.showinfo(title="提示", message="您未选择需要打开的 Wine 容器")
return
tkinter.messagebox.showinfo(title="提示", message="如果安装字体?只需要把字体文件复制到此字体目录\n按下“OK”按钮可以打开字体目录")
os.system("xdg-open \"" + e1.get().replace("\"", "\\\"") + "/drive_c/windows/Fonts\"")
def UninstallProgram():
threading.Thread(target=UninstallProgram_threading).start()
def UninstallProgram_threading():
DisableButton(True)
if e1.get() == "": # 判断文本框是否有内容
tkinter.messagebox.showinfo(title="提示", message="没有填写需要使用的 wine 容器")
DisableButton(False)
else: # 如果都有
#text = subprocess.getoutput("WINEPREFIX='" + e1.get() + "' " + wine[o1_text.get()] + " '" + e2.get() + "'") # 运行
res = subprocess.Popen(["WINEPREFIX='" + e1.get() + "' " + wine[o1_text.get()] + " '" + programPath + "/geek.exe'"], shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
# 清空文本框内容
returnText.config(state=tk.NORMAL)
returnText.delete(1.0, "end")
returnText.config(state=tk.DISABLED)
# 实时读取程序返回
while res.poll() is None:
returnText.config(state=tk.NORMAL)
text = res.stdout.readline().decode("utf8")
returnText.insert("end", text)
print(text)
returnText.config(state=tk.DISABLED)
DisableButton(False)
2021-05-23 15:54:04 +08:00
###########################
# 加载配置
###########################
if not os.path.exists(get_home() + "/.config/deepin-wine-runner"): # 如果没有配置文件夹
os.mkdir(get_home() + "/.config/deepin-wine-runner") # 创建配置文件夹
if not os.path.exists(get_home() + "/.config/deepin-wine-runner/ShellHistory.json"): # 如果没有配置文件
write_txt(get_home() + "/.config/deepin-wine-runner/ShellHistory.json", json.dumps({})) # 创建配置文件
if not os.path.exists(get_home() + "/.config/deepin-wine-runner/FindExeHistory.json"): # 如果没有配置文件
write_txt(get_home() + "/.config/deepin-wine-runner/FindExeHistory.json", json.dumps({})) # 创建配置文件
if not os.path.exists(get_home() + "/.config/deepin-wine-runner/WineBottonHistory.json"): # 如果没有配置文件
write_txt(get_home() + "/.config/deepin-wine-runner/WineBottonHistory.json", json.dumps({})) # 创建配置文件
if not os.path.exists(get_home() + "/.config/deepin-wine-runner/FindExe.json"): # 如果没有配置文件
write_txt(get_home() + "/.config/deepin-wine-runner/FindExe.json", json.dumps({"path": "~"})) # 写入(创建)一个配置文件
if not os.path.exists(get_home() + "/.config/deepin-wine-runner/WineBotton.json"): # 如果没有配置文件
write_txt(get_home() + "/.config/deepin-wine-runner/WineBotton.json", json.dumps({"path": "~/.deepinwine"})) # 写入(创建)一个配置文件
###########################
# 设置变量
###########################
# 如果要添加其他 wine请在字典添加其名称和执行路径
2022-07-01 15:16:00 +08:00
wine = {"deepin-wine": "deepin-wine", "deepin-wine5": "deepin-wine5", "wine": "wine", "wine64": "wine64", "deepin-wine5 stable": "deepin-wine5-stable", "deepin-wine6 stable": "deepin-wine6-stable", "spark-wine7-devel": "spark-wine7-devel"}
2021-05-23 15:54:04 +08:00
shellHistory = list(json.loads(readtxt(get_home() + "/.config/deepin-wine-runner/ShellHistory.json")).values())
findExeHistory = list(json.loads(readtxt(get_home() + "/.config/deepin-wine-runner/FindExeHistory.json")).values())
wineBottonHistory = list(json.loads(readtxt(get_home() + "/.config/deepin-wine-runner/WineBottonHistory.json")).values())
###########################
# 程序信息
###########################
2021-07-27 15:05:19 +08:00
iconPath = "{}/icon.png".format(os.path.split(os.path.realpath(__file__))[0])
2021-05-23 15:54:04 +08:00
programUrl = "https://gitee.com/gfdgd-xi/deep-wine-runner"
2022-07-03 18:38:02 +08:00
version = "1.5.0"
2021-05-23 15:54:04 +08:00
goodRunSystem = "Linux"
2022-07-03 18:38:02 +08:00
programPath = os.path.split(os.path.realpath(__file__))[0] # 返回 string
2021-05-23 15:54:04 +08:00
about = '''一个基于 Python3 的 tkinter 制作的 wine 运行器
版本{}
适用平台{}
tkinter 版本{}
程序官网{}
©2020-{} gfdgd xi'''.format(version, goodRunSystem, tk.TkVersion, programUrl, time.strftime("%Y"))
tips = '''提示:
2021-07-27 15:05:19 +08:00
1使用终端运行该程序可以看到 wine 以及程序本身的提示和报错;
2wine 32 位和 64 位的容器互不兼容;
3部分 wine 系统没有预装本程序没有设置任何 wine 的依赖项如果需要使用请自行安装'''
2022-07-03 18:38:02 +08:00
updateThingsString = '''*1、支持显示 wine 程序运行时的返回内容
*2优化打包方式减少从 pip 安装的库并将 pip 源设为阿里源提升下载速度
*3新增 spark-wine7-devel
*4支持从程序启动用于安装 wine 的程序在菜单栏的程序
5优化 wine 安装脚本在安装星火应用商店的 wine 时支持检测是否有 ss-apt-fast如果有就调用替代 apt 提升安装速度
6支持关闭指定 wine 的进程以及访问对应 wine 容器的目录和字体目录
7从生成shell脚本改为升级到desktop文件
2021-07-27 15:05:19 +08:00
'''
2021-05-23 15:54:04 +08:00
title = "wine 运行器 {}".format(version)
2022-07-03 18:38:02 +08:00
updateTime = "2022年07月03日"
2021-05-23 15:54:04 +08:00
updateThings = "{} 更新内容:\n{}\n更新时间:{}".format(version, updateThingsString, updateTime, time.strftime("%Y"))
###########################
# 窗口创建
###########################
2021-07-27 15:05:19 +08:00
win = tk.Tk() # 创建窗口
win.title(title) # 设置标题
window = ttk.Frame()
# 设置变量以修改和获取值项
2021-01-29 12:07:03 +08:00
o1_text = tk.StringVar()
2021-05-23 15:54:04 +08:00
combobox1 = tk.StringVar()
2021-01-29 12:07:03 +08:00
o1_text.set("deepin-wine")
# 创建控件
2022-07-03 18:38:02 +08:00
controlFrame = ttk.Frame(window)
sendFrame = ttk.Frame(window)
button1 = ttk.Button(window, text="浏览", command=liulanbutton) # 创建按钮控件
button2 = ttk.Button(window, text="浏览", command=liulanexebutton) # 创建按钮控件
2022-07-03 18:38:02 +08:00
button3 = ttk.Button(controlFrame, text="启动", command=runexebutton) # 创建按钮控件
killProgram = ttk.Button(controlFrame, text="停止", command=KillProgram)
openWineBotton = ttk.Button(controlFrame, text="打开 Wine 容器所在目录", command=OpenWineBotton)
installWineFont = ttk.Button(controlFrame, text="安装字体", command=OpenWineFontPath)
uninstallProgram = ttk.Button(controlFrame, text="卸载程序", command=UninstallProgram)
button5 = ttk.Button(sendFrame, text="创建用于运行的 desktop 文件到桌面", command=make_desktop_on_desktop) # 创建按钮控件
saveDesktopFileOnLauncher = ttk.Button(sendFrame, text="创建用于运行的 desktop 文件到启动器", command=make_desktop_on_launcher) # 创建按钮控件
label1 = ttk.Label(window, text="选择你想要使用的 wine 容器:") # 创建标签控件
label2 = ttk.Label(window, text="选择要启动的 Windows 应用") # 创建标签控件
label3 = ttk.Label(window, text="选择要使用的 wine 版本") # 创建标签控件
2022-07-03 18:38:02 +08:00
label4 = ttk.Label(window, text="设置标题以便把上方填写的信息写入到desktop文件里") # 创建标签控件
2021-05-23 15:54:04 +08:00
e1 = ttk.Combobox(window, width=100) # 创建文本框控件
e2 = ttk.Combobox(window, width=100) # 创建文本框控件
combobox1 = ttk.Combobox(window, width=100)
2022-07-03 18:38:02 +08:00
o1 = ttk.OptionMenu(window, o1_text, "deepin-wine", *list(wine)) # 创建选择框控件
returnText = tk.Text(window)
2021-07-27 15:05:19 +08:00
menu = tk.Menu(window, background="white") # 设置菜单栏
programmenu = tk.Menu(menu, tearoff=0, background="white") # 设置“程序”菜单栏
2021-03-13 09:55:49 +08:00
menu.add_cascade(label="程序", menu=programmenu)
2022-07-03 18:38:02 +08:00
programmenu.add_command(label="安装 wine", command=InstallWine)
programmenu.add_separator() # 设置分界线
2021-05-23 15:54:04 +08:00
programmenu.add_command(label="清空软件历史记录", command=CleanProgramHistory)
programmenu.add_separator() # 设置分界线
2021-03-13 09:55:49 +08:00
programmenu.add_command(label="退出程序", command=window.quit) # 设置“退出程序”项
2021-07-27 15:05:19 +08:00
help = tk.Menu(menu, tearoff=0, background="white") # 设置“帮助”菜单栏
2021-03-13 09:55:49 +08:00
menu.add_cascade(label="帮助", menu=help)
2021-05-23 15:54:04 +08:00
help.add_command(label="程序官网", command=OpenProgramURL) # 设置“程序官网”项
help.add_separator()
2021-03-13 09:55:49 +08:00
help.add_command(label="小提示", command=helps) # 设置“小提示”项
2021-05-23 15:54:04 +08:00
help.add_command(label="更新内容", command=UpdateThings) # 设置“更新内容”项
2021-03-13 09:55:49 +08:00
help.add_command(label="关于这个程序", command=about_this_program) # 设置“关于这个程序”项
2021-07-27 15:05:19 +08:00
# 设置窗口
win.iconphoto(False, tk.PhotoImage(file=iconPath))
themes = ttkthemes.ThemedStyle(win)
themes.set_theme("adapta")
win.config(bg="white")
2021-05-23 15:54:04 +08:00
# 设置控件
2021-07-27 15:05:19 +08:00
menu.configure(activebackground="white")
programmenu.configure(activebackground="white")
help.configure(activebackground="white")
2021-05-23 15:54:04 +08:00
e1['value'] = findExeHistory
e2['value'] = wineBottonHistory
combobox1['value'] = shellHistory
2022-07-03 18:38:02 +08:00
returnText.insert("end", "此可以查看到 Wine 应用安装时的程序返回值")
returnText.config(state=tk.DISABLED)
# 添加控件
2021-07-27 15:05:19 +08:00
win.config(menu=menu) # 显示菜单栏
label1.grid(row=0, column=0)
label2.grid(row=1, column=0)
label3.grid(row=2, column=0)
label4.grid(row=4, column=0)
e1.grid(row=0, column=1)
e2.grid(row=1, column=1)
2021-05-23 15:54:04 +08:00
#combobox1.grid(row=4, column=1)
combobox1.grid(row=4, column=1)
button1.grid(row=0, column=2)
button2.grid(row=1, column=2)
2022-07-03 18:38:02 +08:00
controlFrame.grid(row=3, column=0, columnspan=3)
button3.grid(row=0, column=0)
killProgram.grid(row=0, column=1)
openWineBotton.grid(row=0, column=2)
installWineFont.grid(row=0, column=3)
uninstallProgram.grid(row=0, column=4)
sendFrame.grid(row=5, column=0, columnspan=3)
button5.grid(row=0, column=0)
saveDesktopFileOnLauncher.grid(row=0, column=1)
o1.grid(row=2, column=1)
2022-07-03 18:38:02 +08:00
returnText.grid(row=6, column=0, columnspan=3)
# 启动窗口
2021-07-27 15:05:19 +08:00
window.pack()
win.mainloop()