#! /usr/bin/python
# -*- coding: utf-8 -*-
"""
This file is part of librelight.
librelight is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 2 of the License, or
(at your option) any later version.
librelight is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with librelight. If not, see .
(c) 2012 micha.rathfelder@gmail.com
"""
import json
import time
import sys
import _thread as thread
import tkinter
import tkinter as tk
from tkinter import font
import lib.chat as chat
import lib.motion as motion
root = tk.Tk()
root["bg"] = "grey" #white
root.title( __file__)
#default_font = font.Font(family='Helvetica', size=12, weight='bold')
Font = font.Font(family='Helvetica', size=9, weight='normal')
FontBold = font.Font(family='Helvetica', size=9, weight='bold')
#default_font.configure(size=9)
root.option_add("*Font", Font)
from collections import OrderedDict
show_name = "GloryCamp2021"
show_name = "GloryCamp2021"
CUES = OrderedDict()
groups = OrderedDict()
BLIND = 0
STORE = 0
POS = ["PAN","TILT","MOTION"]
COLOR = ["RED","GREEN","BLUE","COLOR"]
BEAM = ["GOBO","G-ROT","PRISMA","P-ROT","FOCUS","SPEED"]
INT = ["DIM","SHUTTER","STROBE","FUNC"]
client = chat.tcp_sender()
fade = 1
def build_cmd(dmx,val,args=[fade]):
cmd=""
if type(val) is float or type(val) is int:
cmd += ",d{}:{:0.4f}".format(dmx,val)
else:
cmd += ",d{}:{}".format(dmx,val)
for val in args:
if type(val) is float or type(val) is int:
cmd += ":{:0.4f}".format(val)
else:
cmd += ":{}".format(val)
return cmd
def update_dmx(attr,data):
global BLIND
dmx = data["DMX"]
val = None
cmd=""
if attr == "VDIM":
for attr in data["ATTRIBUT"]:
dmx = data["DMX"]
if data["ATTRIBUT"][attr]["NR"] < 0:
continue
dmx += data["ATTRIBUT"][attr]["NR"]
#print(attr)
val = data["ATTRIBUT"][attr]["VALUE"]
if data["ATTRIBUT"][attr]["MASTER"]:
val = val * (data["ATTRIBUT"]["VDIM"]["VALUE"] / 255.)
if val is not None:
#cmd += ",d{}:{:0.4f}".format(dmx,int(val))
cmd += build_cmd(dmx,val)
#print("cmd",cmd)
elif data["ATTRIBUT"][attr]["NR"] >= 0:
dmx += data["ATTRIBUT"][attr]["NR"]
val = data["ATTRIBUT"][attr]["VALUE"]
if data["ATTRIBUT"][attr]["MASTER"]:
if "VDIM" in data["ATTRIBUT"]:
val = val * (data["ATTRIBUT"]["VDIM"]["VALUE"] / 255.)
if val is not None:
#cmd += ",d{}:{}".format(dmx,int(val))
cmd += build_cmd(dmx,val)
#print("cmd",cmd)
if not BLIND:
#client.send(cmd )
pass
else:
cmd=""
return cmd
class Xevent():
def __init__(self,fix,elem,attr=None,data=None,mode=None):
self.data=data
self.attr = attr
self.elem = elem
self.mode = mode
def encoder(self,fix,attr,data,elem,action=""):
if action == "click":
if self.data["ATTRIBUT"][attr]["ACTIVE"]:
self.data["ATTRIBUT"][attr]["ACTIVE"] = 0
self.elem["bg"] = "grey"
else:
self.data["ATTRIBUT"][attr]["ACTIVE"] = 1
self.elem["bg"] = "yellow"
return 1
v2=data["ATTRIBUT"][attr]["VALUE"]
change=0
increment = 4.11
if action == "+":
v2+= increment
v = "+{:0.4f}".format( increment ) #) #4.11"
change=1
elif action == "-":
v2-= 4.11
v = "-{:0.4f}".format( increment ) #) #4.11"
change=1
if v2 < 0:
v2=0
elif v2 > 256:
v2=256
if change:
data["ATTRIBUT"][attr]["ACTIVE"] = 1
elem["bg"] = "yellow"
#v2 = v
#v = data["ATTRIBUT"][attr]["VALUE"]
data["ATTRIBUT"][attr]["VALUE"] = v2
elem["text"] = "{} {:0.4f}".format(attr,v2)
#worker.fade_dmx(fix,attr,data,v,v2,ft=0)
cmd=update_dmx(attr=attr,data=data)
#data["ATTRIBUT"][attr]["VALUE"] = v2
client.send(cmd)
def cb(self,event):
#print("cb",self,event,data)
print("cb",self.attr,self.mode,event)
#print(self.obj.keys())
try:
#v = self.data["ATTRIBUT"][self.attr]
global STORE
global BLIND
change = 0
if self.mode == "COMMAND":
if self.attr == "CLEAR":
if event.num == 1:
if STORE:
self.data.val_commands["STORE"] = 0
STORE = 0
self.data.elem_commands["STORE"]["bg"] = "lightgrey"
else:
for fix in self.data.fixtures:
print( "clr",fix)
data = self.data.fixtures[fix]
#print("elm",self.data.elem_attr[fix])
for attr in data["ATTRIBUT"]:
if attr.endswith("-FINE"):
continue
self.data.elem_attr[fix][attr]["bg"] = "grey"
data["ATTRIBUT"][attr]["ACTIVE"] = 0
#print(data["ATTRIBUT"])
elif self.attr == "BLIND":
if event.num == 1:
if self.data.val_commands[self.attr]:
self.data.val_commands[self.attr] = 0
BLIND = 0
self.data.elem_commands[self.attr]["bg"] = "lightgrey"
else:
self.data.val_commands[self.attr] = 1
BLIND = 1
self.data.elem_commands[self.attr]["bg"] = "red"
print("BLIND",self.data.val_commands)
elif self.attr == "STORE":
if event.num == 1:
if self.data.val_commands[self.attr]:
self.data.val_commands[self.attr] = 0
STORE = 0
self.data.elem_commands[self.attr]["bg"] = "lightgrey"
else:
self.data.val_commands[self.attr] = 1
STORE = 1
self.data.elem_commands[self.attr]["bg"] = "red"
print("BLIND",self.data.val_commands)
elif self.attr == "BACKUP":
self.data.backup_presets()
return 0
elif self.mode == "INPUT":
print(self.data.entry.get())
if event.keycode == 36:
x=self.data.entry.get()
client.send(x)
#self.data.entry.clean()
#self.data
#chat.send("")
elif self.mode == "INPUT2":
print(self.data.entry2.get())
if event.keycode == 36:
x=self.data.entry2.get()
client.send(x)
#self.data.entry.clean()
elif self.mode == "INPUT3":
print(self.data.entry3.get())
if event.keycode == 36:
x=self.data.entry3.get()
client.send(x)
#self.data.entry.clean()
#self.data
#chat.send("")
elif self.mode == "PRESET":
nr = self.attr #int(self.attr.split(":")[1])-1
if event.num == 1:
if STORE:
print("STORE PRESET")
sdata = {}
for fix in self.data.fixtures:
data = self.data.fixtures[fix]
for attr in data["ATTRIBUT"]:
if data["ATTRIBUT"][attr]["ACTIVE"]:
if fix not in sdata:
sdata[fix] = {}
if attr not in sdata[fix]:
sdata[fix][attr] = data["ATTRIBUT"][attr]["VALUE"]
print(sdata)
self.data.val_presets[nr] = sdata
if len(sdata):
self.data.elem_presets[nr]["bg"] = "yellow"
else:
self.data.elem_presets[nr]["bg"] = "grey"
#self.data.elem_presets[nr].option_add("*Font", FontBold)
label = ""
if nr in self.data.label_presets:
#print(dir(self.data))
label = self.data.label_presets[nr]
self.data.elem_presets[nr]["text"] = "Preset:"+str(nr)+":\n"+str(len(sdata))+":"+label
print(self.data.val_presets)
self.data.val_commands["STORE"] = 0
STORE = 0
self.data.elem_commands["STORE"]["bg"] = "lightgrey"
else:
print("GO PRESET")
if nr not in self.data.val_presets:
self.data.val_presets[nr] = OrderedDict()
sdata = self.data.val_presets[nr]
cmd = ""
for fix in sdata:
for attr in sdata[fix]:
v2 = sdata[fix][attr]
#print(fix,attr,v)
if fix in self.data.fixtures:
#print("==",self.data.fixtures[fix]["ATTRIBUT"])
if attr in self.data.fixtures[fix]["ATTRIBUT"]:
data = self.data.fixtures[fix]
v=self.data.fixtures[fix]["ATTRIBUT"][attr]["VALUE"]
self.data.fixtures[fix]["ATTRIBUT"][attr]["VALUE"] = v2
self.data.elem_attr[fix][attr]["text"] = str(attr)+' '+str(round(v,2))
cmd+=update_dmx(attr,data)
#worker.fade_dmx(fix,attr,data,v,v2)
client.send(cmd )
print(sdata)
if event.num == 3:
if not STORE:
print("GO PRESET 3")
if nr not in self.data.val_presets:
self.data.val_presets[nr] = OrderedDict()
sdata = self.data.val_presets[nr]
cmd=""
for fix in sdata:
for attr in sdata[fix]:
v2 = sdata[fix][attr]
#print(fix,attr,v)
if fix in self.data.fixtures:
#print("==",self.data.fixtures[fix]["ATTRIBUT"])
if attr in self.data.fixtures[fix]["ATTRIBUT"]:
data = self.data.fixtures[fix]
v=self.data.fixtures[fix]["ATTRIBUT"][attr]["VALUE"]
#self.data.fixtures[fix]["ATTRIBUT"][attr]["VALUE"] = v
#print(str(attr)+' '+str(round(v,2)))
#self.data.elem_attr[fix][attr]["text"] = str(attr)+' '+str(round(v,2))
cmd+=update_dmx(attr,data)
print("go",fix,attr,v,v2)
#worker.fade_dmx(fix,attr,data,v,v2,ft=0)
client.send(cmd )
return 0
elif self.mode == "INPUT":
return 0
if self.mode == "ENCODER":
#if self.attr == "VDIM":
# self.attr = "DIM"
for fix in self.data.fixtures:
data = self.data.fixtures[fix]
for attr in data["ATTRIBUT"]:
if attr.endswith("-FINE"):
continue
elem = self.data.elem_attr[fix][attr]
if self.attr != attr:
continue
if event.num == 1:
#self#encoder(attr=attr,data=data,elem=elem,action="click")
data["ATTRIBUT"][attr]["ACTIVE"] = 1
elem["bg"] = "yellow"
if not data["ATTRIBUT"][attr]["ACTIVE"]:
continue
if event.num == 4:
self.encoder(fix=fix,attr=attr,data=data,elem=elem,action="+")
#if attr == "DIM":
# self.encoder(attr="VDIM",data=data,elem=elem,action="+")
elif event.num == 5:
self.encoder(fix=fix,attr=attr,data=data,elem=elem,action="-")
#if attr == "DIM":
# self.encoder(attr="VDIM",data=data,elem=elem,action="-")
return 0
if event.num == 1:
self.encoder(fix=0,attr=self.attr,data=self.data,elem=self.elem,action="click")
elif event.num == 4:
self.encoder(fix=0,attr=self.attr,data=self.data,elem=self.elem,action="+")
elif event.num == 5:
self.encoder(fix=0,attr=self.attr,data=self.data,elem=self.elem,action="-")
except Exception as e:
print("== cb EXCEPT",e)
print("Error on line {}".format(sys.exc_info()[-1].tb_lineno))
#print(self.elem["text"],self.attr,self.data)
def wheel(event,d=None):
print("wheel",event,d)
import copy
class Master():
def __init__(self):
self.load()
self.all_attr =["DIM","VDIM","PAN","TILT"]
self.elem_attr = {}
self.commands =["BLIND","CLEAR","STORE","EDIT","","","","BACKUP","SET","","SELECT","ACTIVATE","","","",]
self.elem_commands = {}
self.val_commands = {}
self.elem_presets = {}
self.load_presets()
for i in range(8*6):
if i not in self.val_presets:
name = "Preset:"+str(i+1)+":\nXYZ"
#self.presets[i] = [i]
self.val_presets[i] = OrderedDict()
self.label_presets[i] = "-"
def load(self):
fixture = OrderedDict()
DATA = OrderedDict()
DATA["DIM"] = {"NR": 0, "MASTER": "1", "MODE": "F", "VALUE": 0.0,"ACTIVE":0}
fix = {"DMX": 20, "UNIVERS": 2, "NAME": "D", "ATTRIBUT": DATA}
fi = copy.deepcopy(fix)
fi["DMX"] = 1
fi["NAME"] = "F1"
fixture["1"] = fi
fi = copy.deepcopy(fix)
fi["DMX"] = 2
fi["NAME"] = "F2"
fixture["2"] = fi
fi = copy.deepcopy(fix)
fi["DMX"] = 3
fi["NAME"] = "F3"
fixture["3"] = fi
fi = copy.deepcopy(fix)
fi["DMX"] = 4
fi["NAME"] = "F4"
fixture["4"] = fi
fi = copy.deepcopy(fix)
fi["DMX"] = 11
fi["NAME"] = "FL"
fixture["11"] = fi
fi = copy.deepcopy(fix)
fi["DMX"] = 24
fi["NAME"] = "P"
fixture["24"] = fi
DATA = OrderedDict()
DATA["DIM"] = {"NR": 0, "MASTER": "1", "MODE": "F", "VALUE": 0.0,"ACTIVE":0}
DATA["RED"] = {"NR": 3, "MASTER": "", "MODE": "F", "VALUE": 255.0,"ACTIVE":0}
DATA["GREEN"] = {"NR": 4, "MASTER": "", "MODE": "F", "VALUE": 255.0,"ACTIVE":0}
DATA["BLUE"] = {"NR": 5, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
fix = {"DMX": 20, "UNIVERS": 2, "NAME": "IRGB", "ATTRIBUT": DATA}
fi = copy.deepcopy(fix)
fi["DMX"] = 401
#fixture["1001"] = fi
fi = copy.deepcopy(fix)
fi["DMX"] = 421
fi["ATTRIBUT"]["BLUE"]["VALUE"] = 22
#fixture["1002"] = fi
fi = copy.deepcopy(fix)
fi["DMX"] = 441
fi["ATTRIBUT"]["BLUE"]["VALUE"] = 22
#fixture["1003"] = fi
DATA = OrderedDict()
DATA["VDIM"] = {"NR": -1, "MASTER": "", "MODE": "F", "VALUE": 0.0,"ACTIVE":0}
DATA["RED"] = {"NR": 2, "MASTER": "1", "MODE": "F", "VALUE": 255.0,"ACTIVE":0}
DATA["GREEN"] = {"NR": 1, "MASTER": "1", "MODE": "F", "VALUE": 255.0,"ACTIVE":0}
DATA["BLUE"] = {"NR": 0, "MASTER": "1", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
fix3 = {"DMX": 20, "UNIVERS": 2, "NAME": "V+RGB", "ATTRIBUT": DATA}
fi = copy.deepcopy(fix3)
fi["DMX"] = 330
#fixture["2001"] = fi
fi = copy.deepcopy(fix3)
fi["DMX"] = 335
#fixture["2002"] = fi
fi = copy.deepcopy(fix3)
fi["DMX"] = 240
#fixture["2003"] = fi
fi = copy.deepcopy(fix3)
fi["DMX"] = 245
#fixture["2004"] = fi
fi = copy.deepcopy(fix3)
fi["DMX"] = 250
#fixture["2005"] = fi
fi = copy.deepcopy(fix3)
fi["DMX"] = 355
#fixture["2006"] = fi
DATA = OrderedDict()
DATA["DIM-FINE"] = {"NR": 8, "MASTER": "", "MODE": "F", "VALUE": 5.0,"ACTIVE":0}
DATA["VDIM"] = {"NR": -1, "MASTER": "", "MODE": "F", "VALUE": 0.0,"ACTIVE":0}
DATA["PAN"] = {"NR": 0, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
DATA["PAN-FINE"] = {"NR": 1, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
DATA["TILT"] = {"NR": 2, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
DATA["TILT-FINE"] = {"NR": 3, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
DATA["RED"] = {"NR": 6, "MASTER": "1", "MODE": "F", "VALUE": 255.0,"ACTIVE":0}
DATA["GREEN"] = {"NR": 7, "MASTER": "1", "MODE": "F", "VALUE": 255.0,"ACTIVE":0}
DATA["BLUE"] = {"NR": 8, "MASTER": "1", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
fixTMH = {"DMX": 20, "UNIVERS": 2, "NAME": "MH-BEAM", "ATTRIBUT": DATA}
fi = copy.deepcopy(fixTMH)
fi["DMX"] = 241
fixture["3001"] = fi
fi = copy.deepcopy(fixTMH)
fi["DMX"] = 261
fixture["3002"] = fi
DATA = OrderedDict()
DATA["DIM"] = {"NR": 17, "MASTER": "1", "MODE": "F", "VALUE": 0.0,"ACTIVE":0}
DATA["PAN"] = {"NR": 0, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
DATA["PAN-FINE"] = {"NR": 1, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
DATA["TILT"] = {"NR": 2, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
DATA["TILT-FINE"] = {"NR": 3, "MASTER": "", "MODE": "F", "VALUE": 127.0,"ACTIVE":0}
DATA["COLOR"] = {"NR": 8, "MASTER": "", "MODE": "S", "VALUE": 0.0,"ACTIVE":0}
DATA["GOBO"] = {"NR": 9, "MASTER": "", "MODE": "S", "VALUE": 0.0,"ACTIVE":0}
DATA["G-ROT"] = {"NR": 8, "MASTER": "", "MODE": "S", "VALUE": 192.0,"ACTIVE":0}
DATA["PRINSMA"] = {"NR": 10, "MASTER": "", "MODE": "S", "VALUE": 0.0,"ACTIVE":0}
DATA["P-ROT"] = {"NR": 11, "MASTER": "", "MODE": "S", "VALUE": 0.0,"ACTIVE":0}
DATA["FOCUS"] = {"NR": 14, "MASTER": "", "MODE": "F", "VALUE": 0.0,"ACTIVE":0}
DATA["ZOOM"] = {"NR": 13, "MASTER": "", "MODE": "F", "VALUE": 0.0,"ACTIVE":0}
DATA["FROST"] = {"NR": 15, "MASTER": "", "MODE": "F", "VALUE": 0.0,"ACTIVE":0}
DATA["CONTROL"] = {"NR": 5, "MASTER": "", "MODE": "S", "VALUE": 5.0,"ACTIVE":0}
fixREUSH = {"DMX": 300, "UNIVERS": 2, "NAME": "RUSH-BEAM", "ATTRIBUT": DATA}
fi = copy.deepcopy(fixREUSH)
fi["DMX"] = 201
fixture["701"] = fi
fi = copy.deepcopy(fixREUSH)
fi["DMX"] = 220
#fixture["702"] = fi
fi = copy.deepcopy(fixREUSH)
fi["DMX"] = 239
#fixture["703"] = fi
fi = copy.deepcopy(fixREUSH)
fi["DMX"] = 258
#fixture["704"] = fi
fi = copy.deepcopy(fixREUSH)
fi["DMX"] = 277
#fixture["705"] = fi
fi = copy.deepcopy(fixREUSH)
fi["DMX"] = 296
#fixture["706"] = fi
self.fixtures = fixture
def load_presets(self):
filename="presets"
d,l = self._load(filename)
self.val_presets = d
self.label_presets = l
def _load(self,filename):
xfname = "show/"+show_name+"/"+str(filename)+".sav"
print("load",xfname)
f = open(xfname,"r")
lines = f.readlines()
f.close()
data = OrderedDict()
labels = OrderedDict()
for line in lines:
key,label,rdata = line.split("\t",2)
key = int(key)
print(xfname,"load",key,label)
jdata = json.loads(rdata,object_pairs_hook=OrderedDict)
data[key] = jdata
labels[key] = label
return data,labels
def backup_presets(self):
filename = "presets"
data = self.val_presets
labels = self.label_presets
self._backup(filename,data,labels)
def _backup(self,filename,data,labels):
#fixture
xfname = "show/"+show_name+"/"+str(filename)+".sav"
print("backup",xfname)
f = open(xfname,"w")
for key in data:
line = data[key]
label = "label"
if key in labels:
label = labels[key]
if label == "Name-"+str(key):
label = ""
print(xfname,"load",key,label,len(line))
f.write(str(key)+"\t"+label+"\t"+json.dumps(line)+"\n")
f.close()
def draw_dim(self,fix,data,c=0,r=0,frame=None):
i=0
if frame is None:
frame = tk.Frame(root,bg="black")
frame.pack(fill=tk.X, side=tk.TOP)
#b = tk.Button(frame,bg="lightblue", text="FIX:"+str(fix)+" "+data["NAME"],width=20)
#b.bind("