green-signals/setup.py

358 lines
12 KiB
Python
Raw Normal View History

2021-04-06 21:04:10 +02:00
#! /bin/python3
import os
import subprocess
import sys
import tkinter as tk
# imporrt the standard to read and write config files
import configparser
#the main window
master=tk.Tk()
#list of required modules
modules=list()
try:
import vlc
except:
modules.append("python-vlc")
2021-05-26 12:17:29 +02:00
# try:
# import crontab
# except:
# modules.append("python-crontab")
2021-04-06 21:04:10 +02:00
#init variables used in config file and some other places
doAutostart=tk.IntVar()
bild_dauer=tk.IntVar()
DEBUG_PREVIEW=tk.IntVar()
localPath=tk.StringVar()
remoteURL=tk.StringVar()
2021-04-07 15:56:54 +02:00
energyMode=tk.IntVar()
energyStart=tk.StringVar()
energyStop=tk.StringVar()
#when setup can't find the conf file, inited is set to True
inited = False
#URL for the Intro-Screens, use as default, if conf is inited
introURL="https://wolke.netzbegruenung.de/s/2TPGWN5FtWYy2d8/download"
2021-04-06 21:04:10 +02:00
#set "dirty" flag to False
#any change should set it to True, so we can ask on quit to save data
isDirty=False
def createAutostart():
'''
create the auto start file gruene_signale.desktop in $HOME/.config/autostart/
also create intermediate paths if the do not exist
when the file already exists, nothing is done
'''
autostartfile = os.path.expanduser("~/.config/autostart/gruene_signale.desktop")
whereami = os.path.dirname(os.path.realpath(__file__))
contents = u"""[Desktop Entry]
2021-04-06 21:04:10 +02:00
Type=Application
Name=Green Signals Autostart
Comment=Starten der Slideshow von Grüne Signale
NoDisplay=false
Exec=sh -c 'cd %(path)s && /bin/python3 gruene_signale.py'
"""
if os.path.isfile(autostartfile) == False:
if os.path.exists(os.path.dirname(autostartfile)) == False:
os.makedirs(os.path.dirname(autostartfile))
file=open(autostartfile,"w")
file.write(contents%{'path':whereami})
file.close()
def removeAutostart():
'''
check if the auto start file exists, and if it does, remove it
'''
autostartfile = os.path.expanduser("~/.config/autostart/gruene_signale.desktop")
if os.path.isfile(autostartfile) == True:
os.remove(autostartfile)
def checkAutostartfile():
'''
check if the auto start file gruene_signale.desktop already exists
returns True, if it does, False if not
'''
autostartfile = os.path.expanduser("~/.config/autostart/gruene_signale.desktop")
return int(os.path.isfile(autostartfile))
def installModules():
global modules
failed=list()
for module in modules:
try:
subprocess.check_call([sys.executable, "-m", "pip", "install", module])
except:
print("install module %s failed" % module)
failed.append(module)
modules=failed
2021-04-12 14:15:42 +02:00
def validateTimeFields(input,newchar,action,name):
if action == "focusout":
if ":" in input:
hour=int(input.split(":")[0])
minute=int(input.split(":")[1])
if hour < 0 or hour > 23 or minute < 0 or minute > 59:
tk.messagebox.showerror("Eingabefehler", u"Bitte geben sie eine gültige Uhrzeit ein oder die Anzahl der Stunden mit vorangestelltem -")
2021-04-12 14:15:42 +02:00
master.nametowidget(name).focus()
return False
else:
print(name)
if ".start" in name:
energyStart.set("%(h)d:%(m)02d" % {'h':hour, 'm':minute})
elif ".stop" in name:
energyStop.set("%(h)d:%(m)02d" % {'h':hour, 'm':minute})
return True
if newchar == "-" and len(input) < 2:
setDirty()
return True
elif len(input)>0 and input[0] == "-" and newchar in "0123456789":
setDirty()
return True
elif newchar not in '01234567890:.':
2021-04-07 15:56:54 +02:00
return False
setDirty()
return True;
def setDirty(ign=None):
2021-04-06 21:04:10 +02:00
global isDirty
isDirty = True
2021-04-07 15:56:54 +02:00
return True
2021-04-06 21:04:10 +02:00
def readConfig(configFile=None):
2021-04-07 15:56:54 +02:00
global bild_dauer, DEBUG_PREVIEW, localPath, remoteURL, isDirty, inited
global energyStart, energyStop, energyMode
2021-04-06 21:04:10 +02:00
if configFile == None:
configFile = os.path.dirname(os.path.realpath(__file__)) + os.sep + "gruene_signale.conf"
2021-04-07 15:56:54 +02:00
if os.path.exists(configFile) == False:
inited = True
2021-04-06 21:04:10 +02:00
config = configparser.ConfigParser()
config.read(configFile)
2021-04-07 15:56:54 +02:00
#read bild - dauer
2021-04-06 21:04:10 +02:00
try:
bild_dauer.set(int(config.get('bilder', 'dauer')))
except:
bild_dauer.set(10)
isDirty = True
2021-04-07 15:56:54 +02:00
#read debug - preview
2021-04-06 21:04:10 +02:00
try:
val = config.get('debug', 'preview')
except:
val = "0"
if val == "1":
DEBUG_PREVIEW.set(1)
2021-04-07 15:56:54 +02:00
#read pfad - lokal
2021-04-06 21:04:10 +02:00
try:
val = os.path.realpath(config.get('pfad', 'lokal'))
except:
val = os.path.dirname(os.path.realpath(__file__)) + os.sep + "slideshow"
localPath.set(val)
#try to read the remote URL, if not given, work offline
try:
val = config.get('pfad', 'remote')
remoteURL.set(val)
except:
val = None
remoteURL.set("")
2021-04-07 15:56:54 +02:00
#read energ - mode
try:
energyMode.set(int(config.get('energy','mode')))
except:
energyMode.set(0)
#read energy - start
try:
val = config.get('energy', 'start')
except:
val = ""
energyStart.set(val)
#read energy - stop
try:
val = config.get('energy', 'stop')
except:
val = ""
energyStop.set(val)
2021-04-06 21:04:10 +02:00
def writeConfig(configFile=None):
if configFile == None:
configFile = os.path.dirname(os.path.realpath(__file__)) + os.sep + "gruene_signale.conf"
config = configparser.ConfigParser()
config.read(configFile)
2021-04-07 15:56:54 +02:00
required_sections = ['bilder','filme','pfad','debug','energy']
for sect in required_sections:
if sect not in config.sections():
config.add_section( sect )
2021-04-06 21:04:10 +02:00
config.set('bilder','dauer',"%d" % bild_dauer.get())
config.set('debug', 'preview',"%d" % DEBUG_PREVIEW.get())
config.set('pfad', 'lokal',localPath.get().strip())
config.set('pfad', 'remote',remoteURL.get().strip())
2021-04-07 15:56:54 +02:00
config.set('energy', 'mode', "%d" % energyMode.get())
config.set('energy', 'start', energyStart.get())
config.set('energy', 'stop', energyStop.get())
2021-04-06 21:04:10 +02:00
with open(configFile, 'w') as file:
config.write(file)
def save():
2021-04-07 15:56:54 +02:00
global isDirty,inited
2021-04-06 21:04:10 +02:00
if isDirty == False:
return;
writeConfig()
if checkAutostartfile() == 1 and doAutostart.get() == 0:
removeAutostart()
if checkAutostartfile() == 0 and doAutostart.get() == 1:
createAutostart()
isDirty=False
2021-04-07 15:56:54 +02:00
inited=False
2021-04-06 21:04:10 +02:00
def quit():
2021-04-07 15:56:54 +02:00
# if isDirty == False:
2021-04-06 21:04:10 +02:00
master.destroy()
exit(0)
2021-04-12 14:15:42 +02:00
def buildGUI_1():
#add button or notification if Python modules (see above) needs to be installed
row=tk.Frame(master,bd=1,relief=tk.SUNKEN)
lab=tk.Label(row,text="Python Module",width=30,anchor='w')
if len(modules) == 0:
obj=tk.Label(row,text="installiert",fg="green")
else:
obj=tk.Button(row,text="Installieren",command=installModules,fg="red")
row.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
def buildGUI_2():
#add autostart checkbox
row=tk.Frame(master,bd=1,relief=tk.SUNKEN)
lab=tk.Label(row,text=u"Grüne Signale automatisch starten",width=30,anchor='w')
2021-04-12 14:15:42 +02:00
obj=tk.Checkbutton(row,text="aktiv",variable=doAutostart,command=setDirty)
row.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
def buildGUI_3():
#add entry field for local path
row=tk.Frame(master,bd=1,relief=tk.SUNKEN)
lab=tk.Label(row,text="lokaler Pfad",width=30,anchor='w')
obj=tk.Entry(row,textvariable=localPath,validate="key",validatecommand=(anyEntryCallback, '%P'))
row.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
def buildGUI_4():
#add entry field for remoteURL
row=tk.Frame(master,bd=1,relief=tk.SUNKEN)
lab=tk.Label(row,text=u"Remote-URL (leer für Offline-Modus)",width=30,anchor='w')
2021-04-12 14:15:42 +02:00
if inited == True:
remoteURL.set(introURL)
obj=tk.Entry(row,textvariable=remoteURL,validate="key",validatecommand=(anyEntryCallback, '%P'))
row.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
def buildGUI_5():
#add scale widget for duration
row=tk.Frame(master,bd=1,relief=tk.SUNKEN)
lab=tk.Label(row,text=u"Anzeigedauer für Bilder (Sekunden)",width=30,anchor='w')
2021-04-12 14:15:42 +02:00
obj=tk.Scale(row,from_=5,to=120,variable=bild_dauer,orient=tk.HORIZONTAL,command=setDirty)
row.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
def buildGUI_6():
#MARK: add controls for energy savings
row1=tk.Frame(master,bd=1,relief=tk.SUNKEN)
row1.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
row2=tk.Frame(row1,bd=1,relief=tk.FLAT)
row2.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
row3=tk.Frame(row1,bd=1,relief=tk.FLAT)
row3.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
row4=tk.Frame(row1,bd=1,relief=tk.FLAT)
row4.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
lab=tk.Label(row2,text="Energiesparoptionen",width=30,anchor='w')
obj0=tk.Radiobutton(row2,text="ohne",variable=energyMode,command=setDirty,value=0)
obj1=tk.Radiobutton(row2,text="RasPi ausschalten",variable=energyMode,command=setDirty,value=1)
obj2=tk.Radiobutton(row2,text="Monitor deaktivieren",variable=energyMode,command=setDirty,value=2)
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj2.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
obj1.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
obj0.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
lab=tk.Label(row3,text="",width=30,anchor='w')
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
lab=tk.Label(row3,text="um",width=3,anchor='w')
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj1=tk.Entry(row3,name="start",textvariable=energyStart,validate="all",validatecommand=(timeEntryCallback, '%P','%S', "%V", "%W"))
obj1.pack(side=tk.LEFT,fill=tk.X,padx=dx)
lab=tk.Label(row3,text="bis",width=3,anchor='w')
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj2=tk.Entry(row3,name="stop",textvariable=energyStop,validate="all",validatecommand=(timeEntryCallback, '%P', '%S', "%V", "%W"))
obj2.pack(side=tk.LEFT,fill=tk.X,padx=dx)
def buildGUI_7():
#add checkbox for Debug Preview Mode
row=tk.Frame(master,bd=1,relief=tk.SUNKEN)
lab=tk.Label(row,text="Debug Preview Modus",width=30,anchor='w')
obj=tk.Checkbutton(row,text="aktiv",variable=DEBUG_PREVIEW,command=setDirty)
row.pack(side=tk.TOP,padx=dx,pady=dy,expand=tk.YES,fill=tk.X)
lab.pack(side=tk.LEFT,pady=dy,padx=dx)
obj.pack(side=tk.RIGHT,expand=tk.YES,fill=tk.X,padx=dx)
def buildGUI_8():
#add buttons
row=tk.Frame(master)
saveButton=tk.Button(row,text=u"Übernehmen",command=save,pady=4,padx=dx)
2021-04-12 14:15:42 +02:00
saveButton.pack(side=tk.LEFT,pady=dy,padx=dx)
quitButton=tk.Button(row,text="Beenden",command=quit,pady=4,padx=dx)
quitButton.pack(side=tk.RIGHT,pady=dy,padx=dx)
row.pack(side=tk.BOTTOM)
def buildGUI():
buildGUI_1()
buildGUI_2()
buildGUI_3()
buildGUI_4()
buildGUI_5()
buildGUI_6()
buildGUI_7()
buildGUI_8()
2021-04-06 21:04:10 +02:00
readConfig()
master.minsize(600,300)
master.geometry("800x450+560+300")
master.title(u"Einstellungen für Grüne Signale")
2021-04-12 14:15:42 +02:00
master.option_add('*Dialog.msg.font', 'System 10')
2021-04-06 21:04:10 +02:00
2021-04-07 15:56:54 +02:00
#register callbacks for entry fields
timeEntryCallback = master.register(validateTimeFields)
anyEntryCallback = master.register(setDirty)
#fixed value for pady and padx used by pack() function
dy=2
dx=4
2021-04-12 14:15:42 +02:00
buildGUI()
master.mainloop()