_LibreLightDesk.py 76 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212
  1. #! /usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. """
  4. This file is part of LibreLight.
  5. LibreLight is free software: you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation, either version 2 of the License, or
  8. (at your option) any later version.
  9. LibreLight is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with LibreLight. If not, see <http://www.gnu.org/licenses/>.
  15. (c) 2012 micha.rathfelder@gmail.com
  16. """
  17. import random
  18. rnd_id = str(random.randint(1000,9000))
  19. rnd_id += " Beta 22.01"
  20. try:
  21. xtitle = __file__
  22. except:
  23. xtitle = "__file__"
  24. if "/" in xtitle:
  25. xtitle = xtitle.split("/")[-1]
  26. import sys
  27. sys.stdout.write("\x1b]2;"+str(xtitle)+" "+str(rnd_id)+"\x07") # terminal title
  28. import json
  29. import time
  30. import sys
  31. import _thread as thread
  32. import traceback
  33. import tkinter
  34. import tkinter as tk
  35. from tkinter import font
  36. import tkinter.simpledialog
  37. import lib.chat as chat
  38. import lib.motion as motion
  39. from collections import OrderedDict
  40. show_name = "GloryCamp2021"
  41. #show_name = "JMS"
  42. #show_name = "Dimmer"
  43. CUES = OrderedDict()
  44. groups = OrderedDict()
  45. class Modes():
  46. def __init__(self):
  47. self.modes = {}
  48. self.__cfg = {}
  49. self.__cb = None
  50. def val(self,mode,value=None):
  51. if value is not None:
  52. return self.set(mode,value)
  53. elif mode in self.modes:
  54. return self.modes[mode]
  55. def get(self,mode,value=None):
  56. return slef.val(mode,value)
  57. def __check(self,mode):
  58. if mode not in self.modes:
  59. self.modes[mode] = 0
  60. self.__cfg[mode] = 0
  61. def cfg(self,mode,data={}):
  62. self.__check(mode)
  63. if type(data) is dict:
  64. for k in data:
  65. v = data[k]
  66. if v not in self.__cfg:
  67. self.__cfg[k] = v
  68. return 1
  69. elif type(data) is str:
  70. if data in self.__cfg:
  71. return self.__cfg[data]
  72. def set(self,mode,value):
  73. protected = ["BLIND","CLEAR"]
  74. self.__check(mode)
  75. out = 0
  76. if mode == "CLEAR":
  77. return 1
  78. elif mode == "ESC":
  79. for m in self.modes:
  80. print("ESC",m)
  81. if m != "BLIND":
  82. self.modes[m] = 0
  83. self.callback(m)
  84. out = 1
  85. return 1
  86. elif value:
  87. for m in self.modes:
  88. if m not in protected and mode not in protected and m != mode:
  89. if self.modes[m]:
  90. self.modes[m]= 0
  91. self.callback(m)
  92. if self.modes[mode]:
  93. self.modes[mode] = 0 # value
  94. else:
  95. self.modes[mode] = 1 #value
  96. out = 1
  97. else:
  98. self.modes[mode] = 0 #value
  99. self.callback(mode)
  100. return value
  101. def set_cb(self,cb):
  102. if cb:
  103. self.__cb = cb
  104. def callback(self,mode):
  105. if self.__cb is not None and mode in self.modes:
  106. value = self.modes[mode]
  107. self.__cb(mode=mode,value=value)
  108. modes = Modes()
  109. #modes.val("BLIND", 0)
  110. #modes.modes["BLIND"] = 0
  111. modes.modes["ESC"] = 0
  112. modes.modes["STORE"] = 0
  113. modes.modes["EDIT"] = 0
  114. modes.modes["MOVE"] = 0
  115. modes.modes["FLASH"] = 0
  116. modes.modes["STONY_FX"] = 0
  117. modes.modes["SELECT"] = 0
  118. modes.modes["ACTIVATE"] = 0
  119. modes.modes["CFG-BTN"] = 0
  120. modes.modes["LABEL"] = 0
  121. def xcb(mode,value=None):
  122. print("xcb","MODE CALLBACK",mode,value)
  123. #modes.set_cb(xcb)
  124. POS = ["PAN","TILT","MOTION"]
  125. COLOR = ["RED","GREEN","BLUE","COLOR"]
  126. BEAM = ["GOBO","G-ROT","PRISMA","P-ROT","FOCUS","SPEED"]
  127. INT = ["DIM","SHUTTER","STROBE","FUNC"]
  128. client = chat.tcp_sender()
  129. fade = 2 #2 #0.1 #1.13
  130. fade_on = 1
  131. fx_prm = {"SIZE":20,"SPEED":100,"OFFSET":50,"BASE":"-","START":0}
  132. def build_cmd(dmx,val,args=[fade],flash=0,xpfx="",attr=""):
  133. cmd=""
  134. if xpfx:
  135. pfx=xpfx
  136. elif flash:
  137. pfx ="df"
  138. else:
  139. pfx ="d"
  140. if type(val) is float or type(val) is int:
  141. cmd += ",{}{}:{:0.4f}".format(pfx,dmx,val)
  142. else:
  143. cmd += ",{}{}:{}".format(pfx,dmx,val)
  144. if flash:
  145. cmd += ":0:0"#.format(val)
  146. else:
  147. for val in args:
  148. if type(val) is float or type(val) is int:
  149. cmd += ":{:0.4f}".format(val)
  150. else:
  151. cmd += ":{}".format(val)
  152. if attr:
  153. cmd += ":"+str(attr)
  154. return cmd
  155. def update_raw_dmx(data ,value=None,args=[fade],flash=0,pfx="d",fx=0):
  156. cmd = []
  157. if flash:
  158. pfx += "f"
  159. for row in data:
  160. if fx:
  161. if value is not None:
  162. # z.b. flush off
  163. xcmd = str(value)+":"+row["FX"].split(":",1)[-1]
  164. else:
  165. xcmd = row["FX"]
  166. else:
  167. if row["VALUE"] is None:
  168. xcmd = ""
  169. else:
  170. if value is not None:
  171. if type(value) is float:
  172. xcmd = "{:0.4f}".format(value)
  173. else:
  174. xcmd = "{}".format(value)
  175. else:
  176. xcmd = "{:0.4f}".format(row["VALUE"])
  177. for arg in args:
  178. if type(arg) is float:
  179. xcmd += ":{}".format(arg)
  180. else:
  181. xcmd += ":{:0.4f}".format(arg)
  182. #print( "pack: FIX",row["FIX"],row["ATTR"], xcmd)
  183. #xcmd += ":{}".format(row["ATTR"])
  184. cmd.append( xcmd)
  185. return cmd
  186. def update_dmx(attr,data,value=None,args=[fade],flash=0,pfx=""):
  187. #global modes #BLIND
  188. #print("update_dmx",data)
  189. dmx = data["DMX"]
  190. dmx = (data["UNIVERS"]*512)+data["DMX"]
  191. val = None
  192. cmd=""
  193. try:
  194. if attr == "DIM" and data["ATTRIBUT"][attr]["NR"] < 0: #VDIM
  195. #print( "VDIM")
  196. for attr in data["ATTRIBUT"]:
  197. dmx = (data["UNIVERS"]*512) + data["DMX"]
  198. dmx = data["DMX"]
  199. if data["ATTRIBUT"][attr]["NR"] < 0: #virtual channels
  200. continue
  201. dmx += data["ATTRIBUT"][attr]["NR"]-1
  202. mode = ""
  203. if "MODE" in data["ATTRIBUT"][attr]:
  204. mode = data["ATTRIBUT"][attr]["MODE"]
  205. #print(attr)
  206. val = data["ATTRIBUT"][attr]["VALUE"]
  207. if data["ATTRIBUT"][attr]["MASTER"]:
  208. val = val * (data["ATTRIBUT"]["DIM"]["VALUE"] / 255.)
  209. if val is not None:
  210. #cmd += ",d{}:{:0.4f}".format(dmx,int(val))
  211. if value is not None:
  212. val = value
  213. if mode == "F": #FADE
  214. cmd += build_cmd(dmx,val,args=args,flash=flash,xpfx=pfx,attr=attr)
  215. else:
  216. cmd += build_cmd(dmx,val,args=[0],flash=flash,xpfx=pfx,attr=attr)
  217. #print("cmd",cmd)
  218. elif data["ATTRIBUT"][attr]["NR"] > 0:
  219. dmx += data["ATTRIBUT"][attr]["NR"]-1
  220. val = data["ATTRIBUT"][attr]["VALUE"]
  221. mode = ""
  222. if "MODE" in data["ATTRIBUT"][attr]:
  223. mode = data["ATTRIBUT"][attr]["MODE"]
  224. if data["ATTRIBUT"][attr]["MASTER"]:
  225. #if "VDIM" in data["ATTRIBUT"]:
  226. if "DIM" in data["ATTRIBUT"] and data["ATTRIBUT"]["DIM"]["NR"] < 0: #VDIM
  227. val = val * (data["ATTRIBUT"]["DIM"]["VALUE"] / 255.)
  228. if val is not None:
  229. #cmd += ",d{}:{}".format(dmx,int(val))
  230. if value is not None:
  231. val = value
  232. if mode == "F": #FADE
  233. cmd += build_cmd(dmx,val,args=args,flash=flash,xpfx=pfx,attr=attr)
  234. else:
  235. cmd += build_cmd(dmx,val,args=[0],flash=flash,xpfx=pfx,attr=attr)
  236. #print("cmd",cmd)
  237. if modes.val("BLIND"):
  238. cmd=""
  239. return cmd
  240. except Exception as e:
  241. cprint("== cb EXCEPT",e,color="red")
  242. cprint("Error on line {}".format(sys.exc_info()[-1].tb_lineno),color="red")
  243. cprint(''.join(traceback.format_exception(None, e, e.__traceback__)),color="red")
  244. raise e
  245. class dummy_event():
  246. def __init__(self):
  247. self.num =0
  248. self.type = 4 #press 5 release
  249. self.set_value=-1
  250. gcolor = 1
  251. def cprint(*text,color="blue",space=" ",end="\n"):
  252. if not gcolor:
  253. print(text)
  254. return 0
  255. if color == "green":
  256. txt = '\033[92m'
  257. elif color == "red":
  258. txt = '\033[0;31m\033[1m'
  259. elif color == "yellow":
  260. txt = '\033[93m\033[1m'
  261. elif color == "cyan":
  262. txt = '\033[96m'
  263. else:
  264. txt = '\033[94m'
  265. for t in text:
  266. txt += str(t ) +" "
  267. #HEADER = '\033[95m'
  268. #OKBLUE = '\033[94m'
  269. #OKCYAN = '\033[96m'
  270. #OKGREEN = '\033[92m'
  271. #WARNING = '\033[93m'
  272. #FAIL = '\033[91m'
  273. #ENDC = '\033[0m'
  274. #BOLD = '\033[1m'
  275. #UNDERLINE = '\033[4m'
  276. txt += '\033[0m'
  277. print(txt,end=end)
  278. #return txt
  279. cprint("________________________________")
  280. class Xevent():
  281. def __init__(self,fix,elem,attr=None,data=None,mode=None):
  282. self.fix = fix
  283. self.data=data
  284. self.attr = attr
  285. self.elem = elem
  286. self.mode = mode
  287. def encoder(self,fix,attr,action="",xfade=None):
  288. cprint(self.fix,fix,attr,color="red")
  289. v = FIXTURES.encoder(fix,attr,action=action,xfade=xfade)
  290. if self.fix:
  291. self.elem["bg"] = "yellow"
  292. self.elem["text"] = "{} {:0.02f}".format(attr,v)
  293. else:
  294. #refresh_gui(self):
  295. pass
  296. def clear(self,event=None):
  297. ok = self.data.FIXTURES.clear(event)
  298. if ok:
  299. for fix in self.data.elem_attr:
  300. for attr in self.data.elem_attr[fix]:
  301. print(type(self.data.elem_attr[fix]))
  302. print(type(self.data.elem_attr[fix][attr]))
  303. print((self.data.elem_attr[fix][attr]))
  304. self.data.elem_attr[fix][attr]["bg"] = "grey"
  305. return 0
  306. if modes.val("STORE"):
  307. self.data.val_commands["STORE"] = 0
  308. modes.val("STORE",0)# = 0
  309. else:
  310. for fix in self.data.FIXTURES.fixtures:
  311. data = self.data.FIXTURES.fixtures[fix]
  312. for attr in data["ATTRIBUT"]:
  313. if attr.endswith("-FINE"):
  314. continue
  315. self.data.elem_attr[fix][attr]["bg"] = "grey"
  316. print( "CB CLEAR" )
  317. def command(self,event):
  318. if self.mode == "COMMAND":
  319. if self.attr == "CLEAR":
  320. if event.num == 1:
  321. self.clear()
  322. #self.button_refresh("STORE","grey")
  323. #modes.val("STORE",0)
  324. modes.val(self.attr,0)
  325. #modes.val("CLEAR",0)
  326. elif self.attr.startswith("SZ:"):#SIN":
  327. #global fx_prm
  328. k = "SIZE"
  329. if event.num == 1:
  330. pass
  331. elif event.num == 2:
  332. pass
  333. elif event.num == 4:
  334. if fx_prm[k] <= 0:
  335. fx_prm[k] = 1
  336. fx_prm[k] *=1.2
  337. elif event.num == 5:
  338. fx_prm[k] /=1.2
  339. #fx_prm[k] =int(fx_prm[k])
  340. if fx_prm[k] > 4000:
  341. fx_prm[k] = 4000
  342. if fx_prm[k] < 0:
  343. fx_prm[k] =0
  344. self.data.elem_fx_commands[self.attr]["text"] = "SZ:{:0.0f}".format(fx_prm[k])
  345. elif self.attr.startswith("SP:"):#SIN":
  346. #global fx_prm
  347. k = "SPEED"
  348. if event.num == 1:
  349. pass
  350. elif event.num == 2:
  351. pass
  352. elif event.num == 4:
  353. if fx_prm[k] <= 0:
  354. fx_prm[k] = 1
  355. fx_prm[k] *=1.2
  356. elif event.num == 5:
  357. fx_prm[k] /=1.2
  358. #fx_prm[k] =int(fx_prm[k])
  359. if fx_prm[k] > 4000:
  360. fx_prm[k] = 4000
  361. if fx_prm[k] < 0:
  362. fx_prm[k] =0
  363. if fx_prm[k] < 0.1:
  364. self.data.elem_fx_commands[self.attr]["text"] = "SP:off".format(fx_prm[k])
  365. else:
  366. self.data.elem_fx_commands[self.attr]["text"] = "SP:{:0.0f}".format(fx_prm[k])
  367. elif self.attr.startswith("ST:"):#SIN":
  368. #global fx_prm
  369. k = "START"
  370. if event.num == 1:
  371. pass
  372. elif event.num == 2:
  373. pass
  374. elif event.num == 4:
  375. if fx_prm[k] <= 0:
  376. fx_prm[k] = 1
  377. fx_prm[k] *=1.2
  378. elif event.num == 5:
  379. fx_prm[k] /=1.2
  380. #fx_prm[k] =int(fx_prm[k])
  381. if fx_prm[k] > 4000:
  382. fx_prm[k] = 4000
  383. if fx_prm[k] < 0:
  384. fx_prm[k] =0
  385. self.data.elem_fx_commands[self.attr]["text"] = "ST:{:0.0f}".format(fx_prm[k])
  386. elif self.attr.startswith("OF:"):#SIN":
  387. #global fx_prm
  388. k = "OFFSET"
  389. if event.num == 1:
  390. pass
  391. elif event.num == 2:
  392. pass
  393. elif event.num == 4:
  394. if fx_prm[k] <= 0:
  395. fx_prm[k] = 1
  396. fx_prm[k] *=1.2
  397. elif event.num == 5:
  398. fx_prm[k] /=1.2
  399. #fx_prm[k] =int(fx_prm[k])
  400. if fx_prm[k] > 1024:
  401. fx_prm[k] = 1024
  402. if fx_prm[k] < 0:
  403. fx_prm[k] =0
  404. self.data.elem_fx_commands[self.attr]["text"] = "OF:{:0.0f}".format(fx_prm[k])
  405. elif self.attr.startswith("BS:"):
  406. k = "BASE"
  407. if event.num == 1:
  408. fx_prm[k] = "0"
  409. elif event.num == 2:
  410. pass
  411. elif event.num == 4:
  412. fx_prm[k] = "+"
  413. elif event.num == 5:
  414. fx_prm[k] = "-"
  415. self.data.elem_fx_commands[self.attr]["text"] = "BS:{}".format(fx_prm[k])
  416. elif self.attr.startswith("FX:"):#SIN":
  417. if event.num == 1:
  418. cmd = ""
  419. offset = 0
  420. offset_flag=0
  421. start = fx_prm["START"]
  422. base = fx_prm["BASE"]
  423. for fix in self.data.FIXTURES.fixtures:
  424. data = self.data.FIXTURES.fixtures[fix]
  425. #print( "ADD FX",fix)
  426. for attr in data["ATTRIBUT"]:
  427. if attr.endswith("-FINE"):
  428. continue
  429. fx=""
  430. if "SIN" in self.attr:
  431. fx = "sinus"
  432. elif "FD" in self.attr:
  433. fx = "fade"
  434. elif "ON2" in self.attr:
  435. fx = "on2"
  436. elif "ON" in self.attr:
  437. fx = "on"
  438. elif "BUM2" in self.attr:
  439. fx = "bump2"
  440. elif "BUM" in self.attr:
  441. fx = "bump"
  442. elif "COS" in self.attr:
  443. fx = "cosinus"
  444. if fx:
  445. if fx_prm["SPEED"] < 0.1:
  446. fx = "off"
  447. else:
  448. if "DIM" in self.attr:
  449. base=""
  450. if attr == "DIM":
  451. if fx_prm["SPEED"] < 0.1:
  452. fx = "off"
  453. else:
  454. fx = "fade"
  455. elif "TILT" in self.attr:
  456. base=""
  457. if attr == "PAN":
  458. fx = "off"
  459. if attr == "TILT":
  460. if fx_prm["SPEED"] < 0.1:
  461. fx = "off"
  462. else:
  463. fx = "sinus"
  464. elif "PAN" in self.attr:
  465. base=""
  466. if attr == "PAN":
  467. if fx_prm["SPEED"] < 0.1:
  468. fx = "off"
  469. else:
  470. fx = "cosinus"
  471. if attr == "TILT":
  472. fx = "off"
  473. elif "CIR" in self.attr:
  474. base=""
  475. if attr == "PAN":
  476. if fx_prm["SPEED"] < 0.1:
  477. fx = "off"
  478. else:
  479. fx = "cosinus"
  480. if attr == "TILT":
  481. if fx_prm["SPEED"] < 0.1:
  482. fx = "off"
  483. else:
  484. fx = "sinus"
  485. if fx:
  486. fx += ":{:0.0f}:{:0.0f}:{:0.0f}:{:0.0f}:{}:".format(fx_prm["SIZE"],fx_prm["SPEED"],start,offset,base)
  487. offset_flag=1
  488. if "FX" not in data["ATTRIBUT"][attr]:
  489. data["ATTRIBUT"][attr]["FX"] =""
  490. print("ADD FX",fix,attr,fx,data["ATTRIBUT"][attr]["ACTIVE"])
  491. if data["ATTRIBUT"][attr]["ACTIVE"] and fx:
  492. print("++ADD FX",fix,attr,fx)
  493. data["ATTRIBUT"][attr]["FX"] = fx #"sinus:40:100:10"
  494. cmd+=update_dmx(attr,data,pfx="fx",value=fx)#,flash=FLASH)
  495. if fx_prm["OFFSET"] > 0.5 and offset_flag:
  496. offset_flag=0
  497. offset += fx_prm["OFFSET"] # add offset on next fixture
  498. #print("offset",offset)
  499. if cmd and not modes.val("BLIND"):
  500. client.send(cmd)
  501. elif self.attr == "FX OFF":
  502. if event.num == 1:
  503. client.send("fx0:alloff:,fxf:alloff:")
  504. self.data.elem_fx_commands[self.attr]["bg"] = "magenta"
  505. for fix in self.data.FIXTURES.fixtures:
  506. data = self.data.FIXTURES.fixtures[fix]
  507. for attr in data["ATTRIBUT"]:
  508. data["ATTRIBUT"][attr]["FX"] = ""
  509. elif self.attr == "FADE":
  510. global fade
  511. global fade_on
  512. if fade < 0.01:
  513. fade = 0.01
  514. elif fade > 100.0:
  515. fade = 100
  516. if event.num == 4:
  517. fade *= 1.1
  518. elif event.num == 5:
  519. fade /= 1.1
  520. elif event.num == 1:
  521. if fade_on:
  522. fade_on = 0
  523. self.data.elem_commands[self.attr]["bg"] = "grey"
  524. else:
  525. fade_on = 1
  526. self.data.elem_commands[self.attr]["bg"] = "green"
  527. elif event.num == 2:
  528. if fade > 1 and fade < 4:
  529. fade = 4
  530. elif fade > 3 and fade < 6:
  531. fade = 6
  532. elif fade > 5 and fade < 7:
  533. fade = 8
  534. elif fade > 7 and fade < 9:
  535. fade = 10
  536. elif fade > 9:
  537. fade = 0.01
  538. elif fade < 1:
  539. fade = 1.1
  540. self.data.elem_commands[self.attr]["text"] = "Fade{:0.2f}".format(fade)
  541. elif self.attr == "BACKUP":
  542. modes.val(self.attr,1)
  543. self.data.PRESETS.backup_presets()
  544. self.data.FIXTURES.backup_patch()
  545. #time.sleep(1)
  546. modes.val(self.attr,0)
  547. else:
  548. if event.num == 1:
  549. print("ELSE",self.attr)
  550. modes.val(self.attr,1)
  551. return 0
  552. def cb(self,event):
  553. #print("cb",self,event,data)
  554. cprint("EVENT cb",self.attr,self.mode,event,color='yellow')
  555. print(["type",event.type,"num",event.num])
  556. #print(dir(event.type))
  557. #print(dir(event),[str(event.type)])#.keys())
  558. try:
  559. #v = self.data["ATTRIBUT"][self.attr]
  560. #global modes
  561. #global STORE
  562. #global BLIND
  563. #global FLASH
  564. #global STONY_FX
  565. #global LABEL
  566. #global SELECT
  567. #global ACTIVATE
  568. #global CFG-BTN
  569. change = 0
  570. if "keysym" in dir(event):
  571. if "Escape" == event.keysym:
  572. self.clear()
  573. #CLEAR
  574. return 0
  575. if self.mode == "COMMAND":
  576. self.command(event)
  577. elif self.mode == "ROOT":
  578. if event.keysym=="Escape":
  579. pass
  580. #STORE = 0
  581. #LABEL = 0
  582. elif self.mode == "INPUT":
  583. print("INP",self.data.entry.get())
  584. if event.keycode == 36:
  585. x=self.data.entry.get()
  586. client.send(x)
  587. #self.data.entry.clean()
  588. #self.data
  589. #chat.send("")
  590. elif self.mode == "INPUT2":
  591. print("INP2",self.data.entry2.get())
  592. if event.keycode == 36:
  593. x=self.data.entry2.get()
  594. client.send(x)
  595. #self.data.entry.clean()
  596. elif self.mode == "INPUT3":
  597. print("INP3",self.data.entry3.get())
  598. if event.keycode == 36:
  599. x=self.data.entry3.get()
  600. client.send(x)
  601. #self.data.entry.clean()
  602. #self.data
  603. #chat.send("")
  604. elif self.mode == "PRESET":
  605. nr = self.attr #int(self.attr.split(":")[1])-1
  606. #print( "RRR", [str(event.type) , event.type] )
  607. #print( "PRESET EVENT",event.num)
  608. if event.num == 1:
  609. if str(event.type) == '4': #4 ButtonPress
  610. #if str(event.type) == "ButtonRelease" or event.type == '5':
  611. if modes.val("STORE"):
  612. self.data.preset_store(nr)
  613. modes.val("STORE",0)
  614. elif modes.val("CFG-BTN"):
  615. _label = self.data.PRESETS.btn_cfg(nr)
  616. txt = tkinter.simpledialog.askstring("CFG-BTN","GO,FLASH,TOGGLE,SWOP\n EXE:"+str(nr+1),initialvalue=_label)
  617. if txt:
  618. self.data.PRESETS.btn_cfg(nr,txt)
  619. self.data.elem_presets[nr]["text"] = self.data.PRESETS.get_btn_txt(nr)
  620. modes.val("CFG-BTN",0)
  621. #self.data.elem_commands["CFG-BTN"]["bg"] = "grey"
  622. elif modes.val("LABEL"):#else:
  623. _label = self.data.PRESETS.label(nr)
  624. txt = tkinter.simpledialog.askstring("CFG-BTN","GO,FLASH,TOGGLE,SWOP\n EXE:"+str(nr+1),initialvalue=_label)
  625. if txt:
  626. self.data.PRESETS.label(nr,txt)
  627. self.data.elem_presets[nr]["text"] = self.data.PRESETS.get_btn_txt(nr)
  628. modes.val("LABEL", 0)
  629. elif modes.val("ACTIVATE"):
  630. self.data.preset_select(nr)
  631. self.data.preset_go(nr,xfade=0,event=event)
  632. modes.val("ACTIVATE", 0)
  633. self.data.elem_commands["ACTIVATE"]["bg"] = "lightgrey"
  634. elif modes.val("SELECT"):
  635. self.data.preset_select(nr)
  636. else:
  637. self.data.preset_go(nr,event=event)
  638. else:
  639. self.data.preset_go(nr,event=event)
  640. if event.num == 3:
  641. if not modes.val("STORE"):
  642. self.data.preset_go(nr,xfade=0,event=event)
  643. return 0
  644. elif self.mode == "INPUT":
  645. return 0
  646. if self.mode == "ENCODER":
  647. #if self.attr == "VDIM":
  648. # self.attr = "DIM"
  649. for fix in FIXTURES.fixtures:
  650. data = FIXTURES.fixtures[fix]
  651. for attr in data["ATTRIBUT"]:
  652. if attr.endswith("-FINE"):
  653. continue
  654. elem = self.data.elem_attr[fix][attr]
  655. if self.attr != attr:
  656. continue
  657. if event.num == 1:
  658. data["ATTRIBUT"][attr]["ACTIVE"] = 1
  659. elem["bg"] = "yellow"
  660. if "FX" in data["ATTRIBUT"][attr]:#["FX"]:# = 1
  661. if data["ATTRIBUT"][attr]["FX"]:# = 1
  662. elem["fg"] = "blue"
  663. else:
  664. elem["fg"] = "blue"
  665. elem["fg"] = "black"
  666. if not data["ATTRIBUT"][attr]["ACTIVE"]:
  667. continue
  668. if event.num == 4:
  669. self.encoder(fix=fix,attr=attr,action="+")
  670. elif event.num == 5:
  671. self.encoder(fix=fix,attr=attr,action="-")
  672. if "set_value" in dir(event) and event.set_value >=0:
  673. print("ENCODER set_value and set_fade",event)
  674. if "set_fade" in dir(event) and event.set_fade >0:
  675. self.encoder(fix=fix,attr=attr,action=event.set_value,xfade=1)
  676. else:
  677. self.encoder(fix=fix,attr=attr,action=event.set_value)
  678. return 0
  679. action=""
  680. if event.num == 1:
  681. action="click"
  682. elif event.num == 4:
  683. action="+"
  684. elif event.num == 5:
  685. action="-"
  686. if action:
  687. self.encoder(fix=self.fix,attr=self.attr,action=action)
  688. except Exception as e:
  689. cprint("== cb EXCEPT",e,color="red")
  690. cprint("Error on line {}".format(sys.exc_info()[-1].tb_lineno),color="red")
  691. cprint(''.join(traceback.format_exception(None, e, e.__traceback__)),color="red")
  692. def wheel(event,d=None):
  693. print("wheel",event,d)
  694. import copy
  695. class Element():
  696. def __init__(self):
  697. self.__data = {}
  698. def set(self,key,val):
  699. self.__data[key] = val
  700. class Base():
  701. def __init__(self):
  702. pass
  703. def _load(self,filename):
  704. xfname = "show/"+show_name+"/"+str(filename)+".sav"
  705. print("load",xfname)
  706. f = open(xfname,"r")
  707. lines = f.readlines()
  708. f.close()
  709. data = OrderedDict()
  710. labels = OrderedDict()
  711. for line in lines:
  712. key,label,rdata = line.split("\t",2)
  713. key = int(key)
  714. #print(xfname,"load",key,label)
  715. #print(line)
  716. jdata = json.loads(rdata,object_pairs_hook=OrderedDict)
  717. nrnull = 0
  718. if "ATTRIBUT" in jdata: # translate old FIXTURES.fixtures start with 0 to 1
  719. for attr in jdata["ATTRIBUT"]:
  720. if "NR" in jdata["ATTRIBUT"][attr]:
  721. nr = jdata["ATTRIBUT"][attr]["NR"]
  722. if nr == 0:
  723. nrnull = 1
  724. break
  725. if nrnull:
  726. print("DMX NR IS NULL",attr,"CHANGE +1")
  727. for attr in jdata["ATTRIBUT"]:
  728. if "NR" in jdata["ATTRIBUT"][attr]:
  729. nr = jdata["ATTRIBUT"][attr]["NR"]
  730. if nr >= 0:
  731. jdata["ATTRIBUT"][attr]["NR"] +=1
  732. data[key] = jdata
  733. labels[key] = label
  734. return data,labels
  735. def _backup(self,filename,data,labels):
  736. #fixture
  737. xfname = "show/"+show_name+"/"+str(filename)+".sav"
  738. print("backup",xfname)
  739. f = open(xfname,"w")
  740. for key in data:
  741. line = data[key]
  742. #print(line)
  743. label = "label"
  744. if key in labels:
  745. label = labels[key]
  746. if label == "Name-"+str(key):
  747. label = ""
  748. #print(xfname,"load",key,label,len(line))
  749. f.write( "{}\t{}\t{}\n".format( key,label,json.dumps(line) ) )
  750. f.close()
  751. class Event():
  752. def __init__(self,name):
  753. self.name=name
  754. print("init",self)
  755. def event(self,event):
  756. print(self.name,event)
  757. class scroll():
  758. def __init__(self,canvas):
  759. self.canvas=canvas
  760. def config(self,event):
  761. canvas = self.canvas
  762. canvas.configure(scrollregion=canvas.bbox("all"))#,width=400,height=200)
  763. def hex_to_rgb(hex):
  764. return tuple(int(hex[i:i+2], 16) for i in (0, 2, 4))
  765. class cb():
  766. def __init__(self,win):
  767. self.win = win
  768. def _callback(self,event):
  769. clobj=event.widget
  770. ## undermouse=find_withtag(master.CURRENT)
  771. undermouse=self.win.find_closest(self.win.CURRENT)
  772. print( repr(undermouse))
  773. def callback(self,event):
  774. print(__file__,self,"callback",event)
  775. cnv = self.win
  776. item = cnv.find_closest(cnv.canvasx(event.x), cnv.canvasy(event.y))[0]
  777. tags = cnv.gettags(item)
  778. #cnv.itemconfigure(self.tag, text=tags[0])
  779. print(tags,item)
  780. color = cnv.itemcget(item, "fill")
  781. cnv.itemconfig("all", width=1)#filla="green")
  782. cnv.itemconfig(item, width=3)#filla="green")
  783. print(color)
  784. print( hex_to_rgb(color[1:]))
  785. class GUI(Base):
  786. def __init__(self):
  787. super().__init__()
  788. self.load()
  789. self.all_attr =["DIM","PAN","TILT"]
  790. self.elem_attr = {}
  791. self.fx_commands =["STONY_FX","FX OFF","\n"
  792. ,"FX:CIR","FX:PAN","FX:TILT","FX:DIM","\n"
  793. ,"SZ:","SP:","ST:","OF:","BS:-","\n"
  794. , "FX:SIN","FX:COS","FX:BUM","FX:BUM2","FX:FD","FX:ON","FX:ON2" ]
  795. self.commands =["\n","ESC","CFG-BTN","LABEL","BACKUP","\n"
  796. ,"SET","SELECT","ACTIVATE","FLASH","FADE","\n"
  797. ,"BLIND","CLEAR","STORE","EDIT","MOVE","\n"
  798. ]
  799. self.elem_fx_commands = {}
  800. self.val_fx_commands = {}
  801. self.elem_commands = {}
  802. self.val_commands = {}
  803. self.elem_presets = {}
  804. self.PRESETS = Presets()
  805. global PRESETS
  806. PRESETS =self.PRESETS
  807. self.PRESETS.load_presets()
  808. self.FIXTURES = Fixtures()
  809. global FIXTURES
  810. FIXTURES = self.FIXTURES
  811. self.FIXTURES.load_patch()
  812. for i in range(8*8*8):
  813. if i not in self.PRESETS.val_presets:
  814. name = "Preset:"+str(i+1)+":\nXYZ"
  815. #self.presets[i] = [i]
  816. self.PRESETS.val_presets[i] = OrderedDict() # FIX
  817. self.PRESETS.val_presets[i]["CFG"] = OrderedDict() # CONFIG
  818. self.PRESETS.label_presets[i] = "-"
  819. modes.set_cb(self.xcb)
  820. def button_refresh(self,name,color,fg=None):
  821. cprint("button_refresh",name,color)
  822. if name in self.elem_commands:
  823. self.elem_commands[name]["bg"] = color
  824. self.elem_commands[name].config(activebackground=color)
  825. if fg:
  826. self.elem_commands[name]["fg"] = fg
  827. print(dir(self.elem_commands[name]))
  828. def xcb(self,mode,value=None):
  829. cprint("MODE CALLBACK",mode,value,color="green",end="")
  830. #cprint(self,"xcb","MODE CALLBACK",mode,value,color="green")
  831. if value:
  832. cprint("===== ON ======",color="red")
  833. self.button_refresh(mode,color="red")#,fg="blue")
  834. else:
  835. cprint("===== OFF ======",color="red")
  836. self.button_refresh(mode,color="lightgrey")#,fg="black")
  837. def load(self,fname=""):
  838. pass
  839. def exit(self):
  840. print("__del__",self)
  841. self.PRESETS.backup_presets()
  842. print("********************************************************")
  843. self.FIXTURES.backup_patch()
  844. print("*********del",self,"***********************************************")
  845. def refresh_gui(self):
  846. for fix in self.FIXTURES.fixtures:
  847. sdata = self.FIXTURES.fixtures[fix]
  848. for attr in sdata["ATTRIBUT"]:
  849. if "FINE" in attr:
  850. continue
  851. v2 = sdata["ATTRIBUT"][attr]["VALUE"]
  852. if fix in self.elem_attr:
  853. elem = self.elem_attr[fix][attr]
  854. #print( attr,v2)
  855. elem["text"] = "{} {:0.2f}".format(attr,v2)
  856. if sdata["ATTRIBUT"][attr]["ACTIVE"]:
  857. elem["bg"] = "yellow"
  858. else:
  859. elem["bg"] = "grey"
  860. def preset_store(self,nr):
  861. #TODO refactor
  862. print("------- STORE PRESET")
  863. data = self.FIXTURES.get_active()
  864. if modes.val("STONY_FX"):
  865. self.PRESETS.store(nr,data,"STONY_FX")
  866. else:
  867. self.PRESETS.store(nr,data)
  868. #global STORE
  869. #STORE = 0
  870. #self.elem_commands["STORE"]["bg"] = "lightgrey"
  871. #CFG = OrderedDict()
  872. #if "CFG" in self.PRESETS.val_presets[nr]: #["CFG"]
  873. # CFG = self.PRESETS.val_presets[nr]["CFG"]
  874. sdata=data
  875. self.PRESETS.val_presets[nr] = sdata
  876. if len(sdata) > 1:
  877. fx_color = 0
  878. val_color = 0
  879. for fix in sdata:
  880. if fix == "CFG":
  881. continue
  882. #print( "$$$$",fix,sdata[fix])
  883. for attr in sdata[fix]:
  884. if "FX" in sdata[fix][attr]:
  885. if sdata[fix][attr]["FX"]:
  886. fx_color = 1
  887. if "VALUE" in sdata[fix][attr]:
  888. if sdata[fix][attr]["VALUE"] is not None:
  889. val_color = 1
  890. self.elem_presets[nr]["fg"] = "black"
  891. if val_color:
  892. self.elem_presets[nr]["bg"] = "yellow"
  893. if fx_color:
  894. self.elem_presets[nr]["fg"] = "blue"
  895. else:
  896. if fx_color:
  897. self.elem_presets[nr]["bg"] = "cyan"
  898. else:
  899. self.elem_presets[nr]["fg"] = "black"
  900. self.elem_presets[nr]["bg"] = "grey"
  901. #self.elem_presets[nr].option_add("*Font", FontBold)
  902. label = ""
  903. if nr in self.PRESETS.label_presets:
  904. #print(dir(self.data))
  905. label = self.PRESETS.label_presets[nr]
  906. BTN="go"
  907. if "CFG" in sdata:#["BUTTON"] = "GO"
  908. if "BUTTON" in sdata["CFG"]:
  909. BTN = sdata["CFG"]["BUTTON"]
  910. txt = str(nr)+":"+str(BTN)+":"+str(len(sdata)-1)+"\n"+label
  911. self.elem_presets[nr]["text"] = txt
  912. #print("GO CFG ",self.PRESETS.val_presets)
  913. def preset_select(self,nr):
  914. print("SELECT PRESET")
  915. sdata = self.PRESETS.val_presets[nr]
  916. cmd = ""
  917. for fix in sdata:
  918. if fix == "CFG":
  919. continue
  920. for attr in sdata[fix]:
  921. v2 = sdata[fix][attr]["VALUE"]
  922. v2_fx = sdata[fix][attr]["FX"]
  923. #print( self.data.elem_attr)
  924. if fix in self.elem_attr:
  925. elem = self.elem_attr[fix][attr]
  926. FIXTURES.fixtures[fix]["ATTRIBUT"][attr]["ACTIVE"] = 1
  927. elem["bg"] = "yellow"
  928. def preset_go(self,nr,xfade=fade,event=None):
  929. print("GO PRESET FADE",nr)
  930. rdata = PRESETS.get_raw_map(nr)
  931. cfg = PRESETS.get_cfg(nr)
  932. fcmd = FIXTURES.update_raw(rdata)
  933. #virtcmd = self.data.FIXTURES.get_virtual(rdata)
  934. xFLASH = 0
  935. value=None
  936. #xfade = fade
  937. if modes.val("FLASH") or ( "BUTTON" in cfg and cfg["BUTTON"] == "SEL"): #FLASH
  938. self.preset_select(nr)
  939. return 0
  940. elif modes.val("FLASH") or ( "BUTTON" in cfg and cfg["BUTTON"] == "FL"): #FLASH
  941. xFLASH = 1
  942. xfade = 0
  943. if event:
  944. if str(event.type) == "ButtonRelease" or event.type == '5' :
  945. # 4 fix vor ThinkPad / Debian 11
  946. if xFLASH:
  947. value = "off"
  948. vvcmd = update_raw_dmx( rdata ,value,[xfade] )
  949. fxcmd = update_raw_dmx( rdata ,value,[xfade],fx=1)
  950. cmd = []
  951. for vcmd,d in [[vvcmd,"d"],[fxcmd,"fx"]]:
  952. if xFLASH:
  953. d+="f"
  954. for i,v in enumerate(fcmd):
  955. DMX = fcmd[i]["DMX"]
  956. if DMX and vcmd[i]:
  957. xcmd = ",{}{}:{}".format(d,DMX,vcmd[i])
  958. cmd.append( xcmd )
  959. if "VIRTUAL" in fcmd[i]:
  960. for a in fcmd[i]["VIRTUAL"]:
  961. DMX = fcmd[i]["VIRTUAL"][a]
  962. if DMX and vcmd[i]:
  963. xcmd = ",{}{}:{}".format(d,DMX,vcmd[i])
  964. cmd.append( xcmd )
  965. cmd = "".join(cmd)
  966. print("cmd",cmd)
  967. if cmd and not modes.val("BLIND"):
  968. client.send(cmd )
  969. self.refresh_gui()
  970. def draw_dim(self,fix,data,c=0,r=0,frame=None):
  971. Font = font.Font(family='Helvetica', size=9, weight='normal')
  972. FontBold = font.Font(family='Helvetica', size=10, weight='bold')
  973. i=0
  974. if frame is None:
  975. frame = tk.Frame(root,bg="black")
  976. frame.pack(fill=tk.X, side=tk.TOP)
  977. #b = tk.Button(frame,bg="lightblue", text="FIX:"+str(fix)+" "+data["NAME"],width=20)
  978. #b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  979. #b.grid(row=r, column=c, sticky=tk.W+tk.E)
  980. #c+=1
  981. #r+=1
  982. if fix not in self.elem_attr:
  983. self.elem_attr[fix] = {}
  984. for attr in data["ATTRIBUT"]:
  985. if attr not in self.all_attr:
  986. self.all_attr.append(attr)
  987. if attr not in self.elem_attr[fix]:
  988. self.elem_attr[fix][attr] = []
  989. if attr.endswith("-FINE"):
  990. continue
  991. v= data["ATTRIBUT"][attr]["VALUE"]
  992. b = tk.Button(frame,bg="lightblue",font=FontBold, text=""+str(fix)+" "+data["NAME"],width=4)
  993. #b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  994. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  995. c+=1
  996. b = tk.Button(frame,bg="grey",font=FontBold, text=str(attr)+' '+str(round(v,2)),width=6)
  997. self.elem_attr[fix][attr] = b
  998. b.bind("<Button>",Xevent(fix=fix,elem=b,attr=attr,data=data).cb)
  999. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1000. c+=1
  1001. if c >=12:
  1002. c=0
  1003. r+=1
  1004. return c,r
  1005. def draw_patch(self,xframe):
  1006. r=0
  1007. c=0
  1008. frame_dim = xframe
  1009. root = frame_dim
  1010. dim_frame = tk.Frame(root,bg="black")
  1011. dim_frame.pack(fill=tk.X, side=tk.TOP)
  1012. root = frame_patch
  1013. fix_frame = tk.Frame(root,bg="black")
  1014. canvas = tk.Canvas(root)
  1015. def yview(event):
  1016. print("yevent",event)
  1017. print(dir(canvas))
  1018. yyy=20.1
  1019. fix_frame.yview_moveto(yyy)
  1020. fix_frame = tk.Frame(root,bg="black")
  1021. fix_frame.pack(fill=tk.X, side=tk.TOP)
  1022. i=0
  1023. c=0
  1024. r=0
  1025. for fix in self.FIXTURES.fixtures:
  1026. i+=1
  1027. data = self.FIXTURES.fixtures[fix]
  1028. print( fix ,data )
  1029. if 1:
  1030. frame = fix_frame
  1031. b = tk.Button(frame,bg="lightblue", text="FIX:"+str(fix)+" "+data["NAME"],width=20)
  1032. b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  1033. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1034. c+=1
  1035. #r+=1
  1036. if fix not in self.elem_attr:
  1037. self.elem_attr[fix] = {}
  1038. patch = ["DMX","UNIVERS"]
  1039. for k in patch:
  1040. v=data[k]
  1041. b = tk.Button(frame,bg="grey", text=str(k)+' '+str(v),width=8)
  1042. #self.elem_attr[fix][attr] = b
  1043. #b.bind("<Button>",Xevent(fix=fix,elem=b,attr=attr,data=data).cb)
  1044. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1045. c+=1
  1046. if c >=8:
  1047. c=1
  1048. r+=1
  1049. for attr in data["ATTRIBUT"]:
  1050. if attr not in self.all_attr:
  1051. self.all_attr.append(attr)
  1052. if attr not in self.elem_attr[fix]:
  1053. self.elem_attr[fix][attr] = []
  1054. if attr.endswith("-FINE"):
  1055. continue
  1056. v= data["ATTRIBUT"][attr]["VALUE"]
  1057. b = tk.Button(frame,bg="grey", text=str(attr)+' '+str(round(v,2)),width=8)
  1058. self.elem_attr[fix][attr] = b
  1059. #b.bind("<Button>",Xevent(fix=fix,elem=b,attr=attr,data=data).cb)
  1060. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1061. c+=1
  1062. if c >=8:
  1063. c=1
  1064. r+=1
  1065. c=0
  1066. r+=1
  1067. def draw_fix(self,xframe):
  1068. r=0
  1069. c=0
  1070. frame_dim=xframe
  1071. frame_fix=xframe
  1072. root = frame_dim
  1073. dim_frame = tk.Frame(root,bg="black")
  1074. dim_frame.pack(fill=tk.X, side=tk.TOP)
  1075. root = frame_fix
  1076. fix_frame = tk.Frame(root,bg="black")
  1077. fix_frame.pack(fill=tk.X, side=tk.TOP)
  1078. Font = font.Font(family='Helvetica', size=9, weight='normal')
  1079. FontBold = font.Font(family='Helvetica', size=10, weight='bold')
  1080. i=0
  1081. c=0
  1082. r=0
  1083. dim_end=0
  1084. for fix in FIXTURES.fixtures:
  1085. i+=1
  1086. data = FIXTURES.fixtures[fix]
  1087. print( fix ,data )
  1088. if(len(data["ATTRIBUT"].keys()) <= 1):
  1089. c,r=self.draw_dim(fix,data,c=c,r=r,frame=dim_frame)
  1090. else:
  1091. if not dim_end:
  1092. dim_end=1
  1093. c=0
  1094. r=0
  1095. #self._draw_fix(fix,data,root=fix_frame)
  1096. frame = fix_frame
  1097. b = tk.Button(frame,bg="lightblue",font=FontBold, text="FIX:"+str(fix)+" "+data["NAME"],width=20)
  1098. b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  1099. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1100. c+=1
  1101. #r+=1
  1102. if fix not in self.elem_attr:
  1103. self.elem_attr[fix] = {}
  1104. for attr in data["ATTRIBUT"]:
  1105. if attr.endswith("-FINE"):
  1106. continue
  1107. if attr not in self.all_attr:
  1108. self.all_attr.append(attr)
  1109. if attr not in self.elem_attr[fix]:
  1110. self.elem_attr[fix][attr] = ["line1348",fix,attr]
  1111. v= data["ATTRIBUT"][attr]["VALUE"]
  1112. b = tk.Button(frame,bg="grey",font=FontBold, text=str(attr)+' '+str(round(v,2)),width=8)
  1113. self.elem_attr[fix][attr] = b
  1114. b.bind("<Button>",Xevent(fix=fix,elem=b,attr=attr,data=data).cb)
  1115. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1116. c+=1
  1117. if c >=8:
  1118. c=1
  1119. r+=1
  1120. c=0
  1121. r+=1
  1122. def draw_enc(self,xframe):
  1123. root2 = xframe
  1124. i=0
  1125. c=0
  1126. r=0
  1127. frame = tk.Frame(root2,bg="black")
  1128. frame.pack( side=tk.TOP,expand=1,fill="both")
  1129. b = tk.Button(frame,bg="lightblue", text="ENCODER",width=6)
  1130. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1131. c+=1
  1132. for attr in self.all_attr:
  1133. if attr.endswith("-FINE"):
  1134. continue
  1135. v=0
  1136. b = tk.Button(frame,bg="orange", text=str(attr)+'',width=6)
  1137. b.bind("<Button>",Xevent(fix=0,elem=b,attr=attr,data=self,mode="ENCODER").cb)
  1138. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1139. c+=1
  1140. if c >=8:
  1141. c=0
  1142. r+=1
  1143. def draw_fx(self,xframe):
  1144. frame_fx=xframe
  1145. i=0
  1146. c=0
  1147. r=0
  1148. #frame = tk.Frame(root,bg="black")
  1149. #frame.pack(fill=tk.X, side=tk.TOP)
  1150. #b = tk.Label(frame,bg="black", text="------------------------------ ---------------------------------------")
  1151. #b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1152. #r=0
  1153. #frame = tk.Frame(root2,bg="black")
  1154. frame = tk.Frame(frame_fx,bg="black")
  1155. frame.pack(fill=tk.X, side=tk.TOP)
  1156. b = tk.Button(frame,bg="lightblue", text="FX.",width=6)
  1157. #b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  1158. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1159. #r+=1
  1160. c+=1
  1161. for comm in self.fx_commands:
  1162. if comm == "\n":
  1163. c=0
  1164. r+=1
  1165. continue
  1166. v=0
  1167. b = tk.Button(frame,bg="lightgrey", text=str(comm),width=6,height=2)
  1168. if comm not in self.elem_fx_commands:
  1169. self.elem_fx_commands[comm] = b
  1170. self.val_fx_commands[comm] = 0
  1171. b.bind("<Button>",Xevent(fix=0,elem=b,attr=comm,data=self,mode="COMMAND").cb)
  1172. if comm == "BLIND":
  1173. b["bg"] = "grey"
  1174. if comm == "CLEAR":
  1175. b["bg"] = "grey"
  1176. if comm == "STONY_FX":
  1177. b["bg"] = "grey"
  1178. if comm == "FADE":
  1179. b["bg"] = "green"
  1180. if comm == "FX OFF":
  1181. b["bg"] = "magenta"
  1182. if comm == "SZ:":
  1183. b["text"] = "SZ:{:0.0f}".format(fx_prm["SIZE"])
  1184. if comm == "SP:":
  1185. b["text"] = "SP:{:0.0f}".format(fx_prm["SPEED"])
  1186. if comm == "ST:":
  1187. b["text"] = "ST:{:0.0f}".format(fx_prm["START"])
  1188. if comm == "OF:":
  1189. b["text"] = "OF:{:0.0f}".format(fx_prm["OFFSET"])
  1190. if comm == "BS:":
  1191. b["text"] = "BS:{}".format(fx_prm["BASE"])
  1192. if comm:
  1193. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1194. c+=1
  1195. if c >=5:
  1196. c=0
  1197. r+=1
  1198. def draw_command(self,xframe):
  1199. frame_cmd=xframe
  1200. i=0
  1201. c=0
  1202. r=0
  1203. frame = tk.Frame(frame_cmd,bg="black")
  1204. frame.pack(fill=tk.X, side=tk.TOP)
  1205. b = tk.Button(frame,bg="lightblue", text="COMM.",width=6)
  1206. #b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  1207. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1208. #r+=1
  1209. c+=1
  1210. for comm in self.commands:
  1211. if comm == "\n":
  1212. c=0
  1213. r+=1
  1214. continue
  1215. v=0
  1216. b = tk.Button(frame,bg="lightgrey", text=str(comm),width=6,height=2)
  1217. if comm not in self.elem_commands:
  1218. self.elem_commands[comm] = b
  1219. self.val_commands[comm] = 0
  1220. b.bind("<Button>",Xevent(fix=0,elem=b,attr=comm,data=self,mode="COMMAND").cb)
  1221. if comm == "BLIND":
  1222. b["bg"] = "grey"
  1223. if comm == "CLEAR":
  1224. b["bg"] = "grey"
  1225. if comm == "STONY_FX":
  1226. b["bg"] = "grey"
  1227. if comm == "FADE":
  1228. b["bg"] = "green"
  1229. if comm == "FX OFF":
  1230. b["bg"] = "magenta"
  1231. if comm == "SZ:":
  1232. b["text"] = "SZ:{:0.0f}".format(fx_prm["SIZE"])
  1233. if comm == "SP:":
  1234. b["text"] = "SP:{:0.0f}".format(fx_prm["SPEED"])
  1235. if comm == "FADE":
  1236. b["text"] = "FADE:{:0.02f}".format(fade)
  1237. if comm == "ST:":
  1238. b["text"] = "ST:{:0.0f}".format(fx_prm["START"])
  1239. if comm == "OF:":
  1240. b["text"] = "OF:{:0.0f}".format(fx_prm["OFFSET"])
  1241. if comm == "BS:":
  1242. b["text"] = "BS:{}".format(fx_prm["BASE"])
  1243. if comm:
  1244. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1245. c+=1
  1246. if c >=5:
  1247. c=0
  1248. r+=1
  1249. def draw_preset(self,xframe):
  1250. i=0
  1251. c=0
  1252. r=0
  1253. root = xframe
  1254. frame = tk.Frame(root,bg="black")
  1255. frame.pack(fill=tk.X, side=tk.TOP)
  1256. i=0
  1257. for k in self.PRESETS.val_presets:
  1258. if i%(8*8)==0 or i ==0:
  1259. c=0
  1260. b = tk.Label(frame,bg="black", text="X" )
  1261. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1262. r+=1
  1263. c=0
  1264. b = tk.Button(frame,bg="lightblue", text="EXEC " )
  1265. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1266. c+=1
  1267. b = tk.Button(frame,bg="lightblue", text="PAGE " + str(int(i/(8*8))+1) )
  1268. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1269. c+=1
  1270. b = tk.Button(frame,bg="lightblue", text="<NAME>" )
  1271. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1272. r+=1
  1273. c=0
  1274. i+=1
  1275. v=0
  1276. label = ""
  1277. if k in self.PRESETS.label_presets:
  1278. label = self.PRESETS.label_presets[k]
  1279. print([label])
  1280. sdata=self.PRESETS.val_presets[k]
  1281. BTN="go"
  1282. if "CFG" in sdata:#["BUTTON"] = "GO"
  1283. if "BUTTON" in sdata["CFG"]:
  1284. BTN = sdata["CFG"]["BUTTON"]
  1285. txt=str(k+1)+":"+str(BTN)+":"+str(len(sdata)-1)+"\n"+label
  1286. b = tk.Button(frame,bg="grey", text=txt,width=8,height=2)
  1287. b.bind("<Button>",Xevent(fix=0,elem=b,attr=k,data=self,mode="PRESET").cb)
  1288. b.bind("<ButtonRelease>",Xevent(fix=0,elem=b,attr=k,data=self,mode="PRESET").cb)
  1289. if k in self.PRESETS.val_presets and len(self.PRESETS.val_presets[k]) :
  1290. b["bg"] = "yellow"
  1291. sdata = self.PRESETS.val_presets[k]
  1292. if len(sdata) > 1:
  1293. fx_color = 0
  1294. val_color = 0
  1295. for fix in sdata:
  1296. if fix == "CFG":
  1297. continue
  1298. #print( "$$$$",fix,sdata[fix])
  1299. for attr in sdata[fix]:
  1300. if "FX" in sdata[fix][attr]:
  1301. if sdata[fix][attr]["FX"]:
  1302. fx_color = 1
  1303. if "VALUE" in sdata[fix][attr]:
  1304. if sdata[fix][attr]["VALUE"] is not None:
  1305. val_color = 1
  1306. b["fg"] = "black"
  1307. if val_color:
  1308. b["bg"] = "gold"
  1309. if fx_color:
  1310. b["fg"] = "blue"
  1311. else:
  1312. if fx_color:
  1313. b["bg"] = "cyan"
  1314. else:
  1315. b["bg"] = "grey"
  1316. if "SEL" in txt:
  1317. b["fg"] = "black"
  1318. b["bg"] = "blue"
  1319. elif "GO" in txt:
  1320. b["fg"] = "black"
  1321. elif "FL" in txt:
  1322. b["fg"] = "red"
  1323. if k not in self.elem_presets:
  1324. self.elem_presets[k] = b
  1325. #self.PRESETS.val_presets[preset] = 0
  1326. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1327. c+=1
  1328. if c >=8:
  1329. c=0
  1330. r+=1
  1331. def draw_input(self):
  1332. i=0
  1333. c=0
  1334. r=0
  1335. frame = tk.Frame(root2,bg="black")
  1336. frame.pack(fill=tk.X, side=tk.TOP)
  1337. b = tk.Label(frame,bg="black", text="------------------------ ---------------------------------------")
  1338. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1339. r=0
  1340. frame = tk.Frame(root2,bg="black")
  1341. frame.pack(fill=tk.X, side=tk.TOP)
  1342. b = tk.Label(frame, text="send:")
  1343. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1344. c+=1
  1345. b = tk.Entry(frame,bg="grey", text="",width=50)
  1346. self.entry = b
  1347. b.bind("<Button>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT").cb)
  1348. b.bind("<Key>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT").cb)
  1349. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1350. b.insert("end","d0:127,fx241:sinus:50:50:10,fx243:cosinus:50:50:10,d201:127,fx201:sinus:50:300:10")
  1351. r+=1
  1352. b = tk.Entry(frame,bg="grey", text="",width=20)
  1353. self.entry2 = b
  1354. b.bind("<Button>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT2").cb)
  1355. b.bind("<Key>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT2").cb)
  1356. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1357. b.insert("end","d1:0:4")
  1358. r+=1
  1359. b = tk.Entry(frame,bg="grey", text="",width=20)
  1360. self.entry3 = b
  1361. b.bind("<Button>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT3").cb)
  1362. #b.bind("<B1-Motion>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT3").cb)
  1363. b.bind("<Key>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT3").cb)
  1364. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1365. b.insert("end","fx:alloff:::")
  1366. def draw_colorpicker(self,xframe):
  1367. import lib.colorpicker as colp
  1368. e = dummy_event()
  1369. r = Xevent(fix=0,elem=None,attr="RED",data=self,mode="ENCODER") #.cb
  1370. g = Xevent(fix=0,elem=None,attr="GREEN",data=self,mode="ENCODER") #.cb
  1371. b = Xevent(fix=0,elem=None,attr="BLUE",data=self,mode="ENCODER") #.cb
  1372. class _CB():
  1373. def __init__(self):
  1374. self.old_color = (0,0,0)
  1375. def cb(self,event,data):
  1376. if "color" in data and self.old_color != data["color"]:
  1377. self.old_color = data["color"]
  1378. else:
  1379. return 0
  1380. color = data["color"]
  1381. print("PPPPPPPOOOOOORRR")
  1382. print("e",event,data)
  1383. print("e",dir(event))#.keys())
  1384. print("e.num",event.num)
  1385. try:
  1386. print("e.stat",event.state)
  1387. except:pass
  1388. if "color" in data and (event.num == 1 or event.num == 3 or event.num==2 or event.state==256):
  1389. e.num=5
  1390. e.type=1
  1391. cr=-1
  1392. cg=-1
  1393. cb=-1
  1394. if event.num == 1:
  1395. e.set_fade=fade
  1396. cr = color[0]
  1397. cg = color[1]
  1398. cb = color[2]
  1399. elif event.num == 3:
  1400. cr = color[0]
  1401. cg = color[1]
  1402. cb = color[2]
  1403. e.set_fade=-1
  1404. elif event.num == 2:
  1405. e.num=1
  1406. e.type=4
  1407. e.set_value=-1
  1408. elif event.state == 256:
  1409. cr = color[0]
  1410. cg = color[1]
  1411. cb = color[2]
  1412. e.set_fade=-1
  1413. else:
  1414. e.set_fade=-1
  1415. e.set_value=cr#color[0]
  1416. r.cb(e)
  1417. e.set_value=cg#color[1]
  1418. g.cb(e)
  1419. e.set_value=cb#color[2]
  1420. b.cb(e)
  1421. e.set_value=-1
  1422. e.set_fade=-1
  1423. print("PICK COLOR:",data["color"])
  1424. _cb=_CB()
  1425. colp.colorpicker(xframe,width=600,height=100, xcb=_cb.cb)
  1426. return 0
  1427. canvas=tk.Canvas(xframe,width=600,height=100)
  1428. canvas["bg"] = "yellow" #"green"
  1429. canvas.pack()
  1430. # RGB
  1431. x=0
  1432. y=0
  1433. j=0
  1434. d = 20
  1435. for i in range(0,d+1):
  1436. fi = int(i*255/d)
  1437. f = 255-fi
  1438. if i > d/2:
  1439. pass#break
  1440. color = '#%02x%02x%02x' % (f, fi, fi)
  1441. print( "farbe", i*10, j, f,fi,fi,color)
  1442. r = canvas.create_rectangle(x, y, x+20, y+20, fill=color)
  1443. x+=20
  1444. def render(self):
  1445. Xroot.bind("<Key>",Xevent(fix=0,elem=None,attr="ROOT",data=self,mode="ROOT").cb)
  1446. #self.draw_patch()
  1447. #self.draw_fix()
  1448. #input()
  1449. #self.draw_enc()
  1450. #self.draw_command()
  1451. #self.draw_fx()
  1452. self.draw_input()
  1453. #self.draw_preset()
  1454. def ScrollFrame(root,width=50,height=100,bd=1):
  1455. print("ScrollFrame init",width,height)
  1456. aframe=tk.Frame(root,relief=tk.GROOVE)#,width=width,height=height,bd=bd)
  1457. #aframe.place(x=0,y=0)
  1458. aframe.pack(side="left",fill="both",expand=1) #x=0,y=0)
  1459. canvas=tk.Canvas(aframe,width=width-24,height=height)
  1460. canvas["bg"] = "black" #"green"
  1461. bframe=tk.Frame(canvas)#,width=width,height=height)
  1462. bframe["bg"] = "blue"
  1463. scrollbar=tk.Scrollbar(aframe,orient="vertical",command=canvas.yview,width=20)
  1464. canvas.configure(yscrollcommand=scrollbar.set)
  1465. scrollbar.pack(side="right",fill="y")
  1466. canvas.pack(side="left",expand=1,fill="both")
  1467. canvas.create_window((0,0),window=bframe,anchor='nw')
  1468. bframe.bind("<Configure>",scroll(canvas).config)
  1469. canvas.bind("<Button>",Event("XXX").event)
  1470. canvas.bind("<Key>",Event("XXX").event)
  1471. return bframe
  1472. #frame = ScrollFrame(root)
  1473. class GUIHandler():
  1474. def __init__(self):
  1475. pass
  1476. def update(self,fix,attr,args={}):
  1477. #print("GUIHandler",fix,attr,args)
  1478. for i,k in enumerate(args):
  1479. v = args[k]
  1480. #print("GUI-H", i,k,v)
  1481. class Fixtures(Base):
  1482. def __init__(self):
  1483. super().__init__()
  1484. #self.load()
  1485. self.fixtures = OrderedDict()
  1486. self.gui = GUIHandler()
  1487. def load_patch(self):
  1488. filename="patch"
  1489. d,l = self._load(filename)
  1490. self.fixtures = OrderedDict()
  1491. for i in l:
  1492. sdata = d[i]
  1493. for attr in sdata["ATTRIBUT"]:
  1494. sdata["ATTRIBUT"][attr]["ACTIVE"] = 0
  1495. #print("load",filename,sdata)
  1496. #if "CFG" not in sdata:
  1497. # sdata["CFG"] = OrderedDict()
  1498. self.fixtures[str(i)] = sdata
  1499. #self.PRESETS.label_presets = l
  1500. def backup_patch(self):
  1501. filename = "patch"
  1502. data = self.fixtures
  1503. labels = {}
  1504. for k in data:
  1505. labels[k] = k
  1506. self._backup(filename,data,labels)
  1507. def update_raw(self,rdata):
  1508. #print("update_raw",rdata)
  1509. cmd = []
  1510. for i,d in enumerate(rdata):
  1511. xcmd = {"DMX":""}
  1512. #print("fix:",i,d)
  1513. fix = d["FIX"]
  1514. attr = d["ATTR"]
  1515. v2 = d["VALUE"]
  1516. v2_fx = d["FX"]
  1517. if fix not in self.fixtures:
  1518. continue
  1519. sdata = self.fixtures[fix] #shortcat
  1520. ATTR = sdata["ATTRIBUT"]
  1521. sDMX = 0
  1522. if sdata["DMX"] > 0:
  1523. print( sdata)
  1524. sDMX = (sdata["UNIVERS"]*512)+sdata["DMX"]
  1525. #sDMX =sdata["DMX"]
  1526. if attr not in ATTR:
  1527. continue
  1528. if ATTR[attr]["NR"] >= 0:
  1529. DMX = sDMX+ATTR[attr]["NR"]-1
  1530. xcmd["DMX"] = str(DMX)
  1531. else:
  1532. if attr == "DIM" and ATTR[attr]["NR"] < 0:
  1533. xcmd["VIRTUAL"] = {}
  1534. for a in ATTR:
  1535. if ATTR[a]["MASTER"]:
  1536. xcmd["VIRTUAL"][a] = sDMX+ATTR[a]["NR"]-1
  1537. #print( "VIRTUAL",xcmd)
  1538. cmd.append(xcmd)
  1539. v=ATTR[attr]["VALUE"]
  1540. if v2 is not None:
  1541. ATTR[attr]["VALUE"] = v2
  1542. #self.data.elem_attr[fix][attr]["text"] = str(attr)+' '+str(round(v,2))
  1543. text = str(attr)+' '+str(round(v,2))
  1544. self.gui.update(fix,attr,args={"text":text})
  1545. return cmd
  1546. def encoder(self,fix,attr,action="",xfade=None):
  1547. print("FIXTURES.encoder",fix,attr,action,xfade)
  1548. data = self.fixtures[fix]
  1549. if action == "click":
  1550. print("encoder",fix,attr,action,data)
  1551. #cprint(type(self.data))
  1552. if data is dict or data is OrderedDict:
  1553. if "ATTRIBUT" in self.data:
  1554. if attr in data["ATTRIBUT"]:
  1555. if "ACTIVE" in data["ATTRIBUT"][attr]:
  1556. if data["ATTRIBUT"][attr]["ACTIVE"]:
  1557. data["ATTRIBUT"][attr]["ACTIVE"] = 0
  1558. else:
  1559. data["ATTRIBUT"][attr]["ACTIVE"] = 1
  1560. return 1
  1561. v2=data["ATTRIBUT"][attr]["VALUE"]
  1562. change=0
  1563. increment = 4.11
  1564. if action == "+":
  1565. v2+= increment
  1566. v = "+{:0.4f}".format( increment ) #) #4.11"
  1567. change=1
  1568. elif action == "-":
  1569. v2-= increment
  1570. v = "-{:0.4f}".format( increment ) #) #4.11"
  1571. change=1
  1572. elif type(action) is int or type(action) is float:
  1573. #v2-= increment
  1574. #v = "-{:0.4f}".format( increment ) #) #4.11"
  1575. v2 = action
  1576. change=1
  1577. if v2 < 0:
  1578. v2=0
  1579. elif v2 > 256:
  1580. v2=256
  1581. out = {}
  1582. if change:
  1583. data["ATTRIBUT"][attr]["ACTIVE"] = 1
  1584. data["ATTRIBUT"][attr]["VALUE"] = v2
  1585. if xfade:
  1586. cmd=update_dmx(attr=attr,data=data)
  1587. else:
  1588. cmd=update_dmx(attr=attr,data=data,args=[0])
  1589. if cmd and not modes.val("BLIND"):
  1590. client.send(cmd)
  1591. return v2
  1592. def get_active(self):
  1593. print(self,"get_active")
  1594. CFG = OrderedDict()
  1595. sdata = OrderedDict()
  1596. sdata["CFG"] = CFG # OrderedDict()
  1597. sdata["CFG"]["FADE"] = fade
  1598. sdata["CFG"]["DEALY"] = 0
  1599. #sdata["CFG"]["BUTTON"] = "GO"
  1600. for fix in self.fixtures:
  1601. data = self.fixtures[fix]
  1602. for attr in data["ATTRIBUT"]:
  1603. if data["ATTRIBUT"][attr]["ACTIVE"]:
  1604. if fix not in sdata:
  1605. sdata[fix] = {}
  1606. if attr not in sdata[fix]:
  1607. sdata[fix][attr] = OrderedDict()
  1608. if not modes.val("STONY_FX"):
  1609. sdata[fix][attr]["VALUE"] = data["ATTRIBUT"][attr]["VALUE"]
  1610. #sdata[fix][attr]["FADE"] = fade
  1611. else:
  1612. sdata[fix][attr]["VALUE"] = None #data["ATTRIBUT"][attr]["VALUE"]
  1613. if "FX" not in data["ATTRIBUT"][attr]:
  1614. data["ATTRIBUT"][attr]["FX"] =""
  1615. sdata[fix][attr]["FX"] = data["ATTRIBUT"][attr]["FX"]
  1616. return sdata
  1617. def select(self,fix=None,attr=None):
  1618. out = 0
  1619. if fix in self.fixtures:
  1620. data = self.fixtures[fix]
  1621. if attr in data["ATTRIBUT"]:
  1622. data["ATTRIBUT"][attr]["ACTIVE"] = 1
  1623. out = 1
  1624. return 1
  1625. def clear(self,event=None):
  1626. out = 0
  1627. if 1:
  1628. for fix in self.fixtures:
  1629. #print( "clr",fix)
  1630. data = self.fixtures[fix]
  1631. #print("elm",self.data.elem_attr[fix])
  1632. for attr in data["ATTRIBUT"]:
  1633. if attr.endswith("-FINE"):
  1634. continue
  1635. if data["ATTRIBUT"][attr]["ACTIVE"]:
  1636. out +=1
  1637. data["ATTRIBUT"][attr]["ACTIVE"] = 0
  1638. #print(data["ATTRIBUT"])
  1639. print( "CB CLEAR" )
  1640. return out
  1641. class Presets(Base):
  1642. def __init__(self):
  1643. super().__init__()
  1644. #self.load()
  1645. def load_presets(self):
  1646. filename="presets"
  1647. d,l = self._load(filename)
  1648. for i in d:
  1649. sdata = d[i]
  1650. if "CFG" not in sdata:
  1651. sdata["CFG"] = OrderedDict()
  1652. if "FADE" not in sdata["CFG"]:
  1653. sdata["CFG"]["FADE"] = 4
  1654. if "DELAY" not in sdata["CFG"]:
  1655. sdata["CFG"]["DELAY"] = 0
  1656. if "BUTTON" not in sdata["CFG"]:
  1657. sdata["CFG"]["BUTTON"] = "GO"
  1658. self.val_presets = d
  1659. self.label_presets = l
  1660. def backup_presets(self):
  1661. filename = "presets"
  1662. data = self.val_presets
  1663. labels = self.label_presets
  1664. self._backup(filename,data,labels)
  1665. def get_cfg(self,nr):
  1666. if nr not in self.val_presets:
  1667. print(self,"error get_cfg no nr:",nr)
  1668. return {}
  1669. if "CFG" in self.val_presets[nr]:
  1670. return self.val_presets[nr]["CFG"]
  1671. def get_raw_map(self,nr):
  1672. print("get_raw_map",nr)
  1673. if nr not in self.val_presets:
  1674. self.val_presets[nr] = OrderedDict()
  1675. self.val_presets[nr]["VALUE"] = 0
  1676. self.val_presets[nr]["FX"] = ""
  1677. sdata = self.val_presets[nr]
  1678. cmd = ""
  1679. out = []
  1680. dmx=-1
  1681. for fix in sdata:
  1682. if fix == "CFG":
  1683. #print("CFG",nr,sdata[fix])
  1684. continue
  1685. for attr in sdata[fix]:
  1686. x = {}
  1687. #print("RAW",attr)
  1688. x["FIX"] = fix
  1689. x["ATTR"] = attr
  1690. x["VALUE"] = sdata[fix][attr]["VALUE"]
  1691. x["FX"] = sdata[fix][attr]["FX"]
  1692. #x["DMX"] = sdata[fix][attr]["NR"]
  1693. out.append(x)
  1694. return out
  1695. def get_btn_txt(self,nr):
  1696. sdata=self.val_presets[nr]
  1697. BTN="go"
  1698. if "CFG" in sdata:
  1699. if "BUTTON" in sdata["CFG"]:
  1700. BTN = sdata["CFG"]["BUTTON"]
  1701. _label = self.label_presets[nr] # = label
  1702. txt=str(nr+1)+":"+str(BTN)+":"+str(len(sdata)-1)+"\n"+_label
  1703. print("get_btn_txt",nr,[txt])
  1704. return txt
  1705. def btn_cfg(self,nr,txt=None):
  1706. if nr not in self.val_presets:
  1707. return ""
  1708. if type(name) is str:
  1709. if "CFG" not in self.val_presets[nr]:
  1710. self.val_presets[nr]["CFG"] = OrderedDict()
  1711. if "BUTTON" not in self.val_presets[nr]["CFG"]:
  1712. self.val_presets[nr]["CFG"]["BUTTON"] = ""
  1713. self.val_presets[nr]["CFG"]["BUTTON"] = txt
  1714. if self.val_presets[nr]["CFG"]["BUTTON"] is None:
  1715. self.val_presets[nr]["CFG"]["BUTTON"] = ""
  1716. print("EEE", self.val_presets[nr]["CFG"]["BUTTON"] )
  1717. return self.val_presets[nr]["CFG"]["BUTTON"]
  1718. def label(self,nr,txt=None):
  1719. if nr not in self.label_presets:
  1720. return ""
  1721. if type(txt) is str:
  1722. self.label_presets[nr] = txt
  1723. print("set label",nr,[txt])
  1724. print("??? ?? set label",nr,[txt])
  1725. return self.label_presets[nr]
  1726. def store(self,nr,data,arg=""):
  1727. #TODO implement
  1728. print(self,"store()",data,arg)
  1729. self.val_presets[nr] = data
  1730. if not self.label_presets:
  1731. self.label_presets = "Neu"
  1732. #return 0
  1733. class GUI_grid():
  1734. def __init__(self,root,data,title="tilte",width=800):
  1735. self.data = data
  1736. self.frame = tk.Frame(root,bg="black",width=width)
  1737. self.frame.pack(fill=tk.BOTH, side=tk.LEFT)
  1738. r=0
  1739. c=0
  1740. i=1
  1741. for row in data:
  1742. self.b = tk.Button(self.frame,bg="lightblue", text=row["text"],width=11,height=4)
  1743. #self.b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  1744. self.b.grid(row=r, column=c, sticky=tk.W+tk.E)#,anchor="w")
  1745. c+=1
  1746. if c % 8 == 0:
  1747. r+=1
  1748. c=0
  1749. i+=1
  1750. self.frame.pack()
  1751. class BEvent():
  1752. def __init__(self,data,cb):
  1753. self._data = data
  1754. self._cb = cb
  1755. def cb(self,event):
  1756. #print(self,event)
  1757. self._cb(event,self._data)
  1758. class GUI_menu():
  1759. def __init__(self,root,data,title="tilte",width=800):
  1760. global tk
  1761. self.data = data
  1762. self.frame = tk.Frame(root,bg="black",width=width)
  1763. self.frame.pack(fill=tk.BOTH, side=tk.LEFT)
  1764. r=0
  1765. c=0
  1766. i=1
  1767. self.b = tk.Label(self.frame,bg="blue", text="MAIN:MENU",width=13,height=1)
  1768. self.b.grid(row=r, column=c, sticky=tk.W+tk.E)#,anchor="w")
  1769. r+=1
  1770. for row in data:
  1771. #print(i)
  1772. #row = data[i]
  1773. self.b = tk.Button(self.frame,bg="lightblue", text=row["text"],width=13,height=3)
  1774. self.b.bind("<Button>",BEvent({"NR":i,"text":row["text"]},self.callback).cb)
  1775. self.b.grid(row=r, column=c, sticky=tk.W+tk.E)#,anchor="w")
  1776. r+=1
  1777. i+=1
  1778. self.frame.pack()
  1779. def callback(self,event,data={}):
  1780. print(self,event,data)
  1781. window_manager.top(data["text"])# = WindowManager()
  1782. lf_nr = 0
  1783. class GUIWindow():
  1784. def __init__(self,title="tilte",master=0,width=100,height=100,left=None,top=None):
  1785. global lf_nr
  1786. if master:
  1787. #Font = font.Font(family='Helvetica', size=9, weight='normal')
  1788. self.tk = tkinter.Tk()#font=Font) #Toplevel()
  1789. #Font = font.Font(family='Helvetica', size=9, weight='normal')
  1790. #FontBold = font.Font(family='Helvetica', size=10, weight='bold')
  1791. #self.tk.default_font.configure(size=9)
  1792. #self.tk.option_add("*Font", FontBold)
  1793. #self.tk.configure(font=Font)
  1794. else:
  1795. self.tk = tkinter.Toplevel()
  1796. self.tk["bg"] = "black"
  1797. self.tk.bind("<Button>",self.callback)
  1798. self.tk.bind("<Key>",self.callback)
  1799. self.tk.title(""+str(title)+" "+str(lf_nr)+":"+str(rnd_id))
  1800. lf_nr+=1
  1801. #self.tk.geometry("270x600+0+65")
  1802. geo ="{}x{}".format(width,height)
  1803. if left is not None:
  1804. geo += "+{}".format(left)
  1805. if top is not None:
  1806. geo += "+{}".format(top)
  1807. #self._event_clear = Xevent(fix=0,elem=None,attr="CLEAR",data=self,mode="ROOT").cb
  1808. self.tk.geometry(geo)
  1809. def title(self,title=None):
  1810. if title is None:
  1811. return self.tk.title()
  1812. else:
  1813. return self.tk.title(title)
  1814. def show(self):
  1815. pass
  1816. #self.frame.pack()
  1817. def mainloop(self):
  1818. self.tk.mainloop()
  1819. def callback(self,event,data={}):
  1820. print("<GUI>",self,event,data)
  1821. #if "keysym" in dir(event):
  1822. # if "Escape" == event.keysym:
  1823. # e=dummy_event()
  1824. # e.num=1
  1825. # self._event_clear(e)
  1826. class WindowManager():
  1827. def __init__(self):
  1828. self.windows = {}
  1829. self.nr= 0
  1830. self.first=""
  1831. def new(self,w,name=""):
  1832. if not self.first:
  1833. if name:
  1834. self.first = name
  1835. else:
  1836. self.first = str(self.nr)
  1837. w.tk.attributes('-topmost',True)
  1838. if name:
  1839. self.windows[str(name)] = w
  1840. else:
  1841. self.windows[str(self.nr)] = w
  1842. self.nr+=1
  1843. #w.show()
  1844. def mainloop(self):
  1845. self.windows[self.first].mainloop()
  1846. def top(self,name):
  1847. name = str(name)
  1848. if name in self.windows:
  1849. self.windows[name].tk.attributes('-topmost',True)
  1850. self.windows[name].tk.attributes('-topmost',False)
  1851. else:
  1852. print(name,"not in self.windows",self.windows.keys())
  1853. window_manager = WindowManager()
  1854. master =GUI()
  1855. w = GUIWindow("MAIN",master=1,width=130,height=450,left=0,top=65)
  1856. data = []
  1857. #data.append({"text":"COMMAND"})
  1858. data.append({"text":"EXEC"})
  1859. data.append({"text":"DIMMER"})
  1860. data.append({"text":"FIXTURES"})
  1861. #data.append({"text":"PRESET"})
  1862. #data.append({"text":"PATCH"})
  1863. #data.append({"text":"ENCODER"})
  1864. f = GUI_menu(w.tk,data)
  1865. window_manager.new(w)
  1866. name="DIMMER"
  1867. w = GUIWindow(name,master=0,width=800,height=400,left=140,top=65)
  1868. w1 = ScrollFrame(w.tk,width=800,height=400)
  1869. #frame_dim = w1 # w.tk
  1870. #master.draw_dim(w1.tk)
  1871. window_manager.new(w,name)
  1872. name="FIXTURES"
  1873. w = GUIWindow(name,master=0,width=800,height=400,left=140,top=65)
  1874. w1 = ScrollFrame(w.tk,width=800,height=400)
  1875. #frame_fix = w1 #w.tk
  1876. master.draw_fix(w1)#.tk)
  1877. window_manager.new(w,name)
  1878. name="ENCODER"
  1879. ww = GUIWindow(name,master=0,width=800,height=50,left=140,top=500)
  1880. Xroot = ww.tk
  1881. #default_font = font.Font(family='Helvetica', size=12, weight='bold')
  1882. Font = font.Font(family='Helvetica', size=9, weight='normal')
  1883. FontBold = font.Font(family='Helvetica', size=10, weight='bold')
  1884. #default_font.configure(size=9)
  1885. Xroot.option_add("*Font", FontBold)
  1886. w = None
  1887. root = tk.Frame(Xroot,bg="black",width="10px")
  1888. root.pack(fill=tk.BOTH,expand=0, side=tk.LEFT)
  1889. root3 = tk.Frame(Xroot,bg="black",width="20px")
  1890. root3.pack(fill=tk.BOTH,expand=0, side=tk.LEFT)
  1891. root2 = tk.Frame(Xroot,bg="black",width="1px")
  1892. master.draw_enc(root2)
  1893. root2.pack(fill=tk.BOTH,expand=0, side=tk.LEFT)
  1894. #w = GUIWindow("GRID",master=0,width=1000,height=200,left=232,top=65)
  1895. #data = []
  1896. #for i in range(10):
  1897. # data.append({"text":"P {:02}".format(i+1)})
  1898. #w = GUI_grid(w.tk,data)
  1899. #window_manager.new(w)
  1900. name = "COMMAND"
  1901. w = GUIWindow(name,master=0,width=350,height=200,left=950,top=65)
  1902. master.draw_command(w.tk)
  1903. window_manager.new(w,name)
  1904. name="EXEC"
  1905. w = GUIWindow(name,master=0,width=800,height=400,left=140,top=65)
  1906. w1 = ScrollFrame(w.tk,width=800,height=400)
  1907. #frame_exe = w.tk
  1908. master.draw_preset(w1)#w.tk)
  1909. window_manager.new(w,name)
  1910. name="PATCH"
  1911. w = GUIWindow(name,master=0,width=800,height=400,left=140,top=65)
  1912. w1 = ScrollFrame(w.tk,width=800,height=400)
  1913. frame_patch = w1 #w.tk
  1914. window_manager.new(w,name)
  1915. name="FX"
  1916. w = GUIWindow(name,master=0,width=350,height=250,left=950,top=305)
  1917. #frame_fx = w.tk
  1918. master.draw_fx(w.tk)
  1919. window_manager.new(w,name)
  1920. #LibreLightDesk
  1921. name="COLERPICKER"
  1922. w = GUIWindow(name,master=0,width=580,height=100,left=80,top=620)
  1923. master.draw_colorpicker(w.tk)
  1924. window_manager.new(w,name)
  1925. #Xroot = tk.Tk()
  1926. #Xroot["bg"] = "black" #white
  1927. #Xroot.title( xtitle+" "+str(rnd_id) )
  1928. #Xroot.geometry("1024x800+130+65")
  1929. master.render()
  1930. #w = frame_fix #GUIWindow("OLD",master=0,width=800,height=500,left=130,top=65)
  1931. window_manager.new(w,name)
  1932. try:
  1933. #root.mainloop()
  1934. #tk.mainloop()
  1935. window_manager.mainloop()
  1936. finally:
  1937. master.exit()