_LibreLightDesk.py 98 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822
  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, version 2 of the License.
  8. LibreLight is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with LibreLight. If not, see <http://www.gnu.org/licenses/>.
  14. (c) 2012 micha@uxsrv.de
  15. """
  16. import random
  17. rnd_id = str(random.randint(1000,9000))
  18. rnd_id += " Beta 22.02 "
  19. import subprocess
  20. rnd_id += subprocess.check_output(['git', 'rev-parse', '--short', 'HEAD']).decode('ascii').strip()
  21. try:
  22. xtitle = __file__
  23. except:
  24. xtitle = "__file__"
  25. if "/" in xtitle:
  26. xtitle = xtitle.split("/")[-1]
  27. import sys
  28. sys.stdout.write("\x1b]2;"+str(xtitle)+" "+str(rnd_id)+"\x07") # terminal title
  29. import json
  30. import time
  31. import sys
  32. import os
  33. import _thread as thread
  34. import traceback
  35. import tkinter
  36. import tkinter as tk
  37. from tkinter import font
  38. import tkinter.simpledialog
  39. import lib.chat as chat
  40. import lib.motion as motion
  41. from collections import OrderedDict
  42. CUES = OrderedDict()
  43. groups = OrderedDict()
  44. class Modes():
  45. def __init__(self):
  46. self.modes = {}
  47. self.__cfg = {}
  48. self.__cb = None
  49. def val(self,mode,value=None):
  50. if value is not None:
  51. return self.set(mode,value)
  52. elif mode in self.modes:
  53. return self.modes[mode]
  54. def get(self,mode,value=None):
  55. return slef.val(mode,value)
  56. def __check(self,mode):
  57. if mode not in self.modes:
  58. self.modes[mode] = 0
  59. self.__cfg[mode] = 0
  60. def cfg(self,mode,data={}):
  61. self.__check(mode)
  62. if type(data) is dict:
  63. for k in data:
  64. v = data[k]
  65. if v not in self.__cfg:
  66. self.__cfg[k] = v
  67. return 1
  68. elif type(data) is str:
  69. if data in self.__cfg:
  70. return self.__cfg[data]
  71. def set(self,mode,value):
  72. protected = ["BLIND","CLEAR","STONY_FX"]
  73. self.__check(mode)
  74. out = 0
  75. if mode == "CLEAR":
  76. return 1
  77. elif mode == "ESC":
  78. for m in self.modes:
  79. print("ESC",m)
  80. if m == "COPY":
  81. PRESETS.clear_copy()
  82. if m == "MOVE":
  83. PRESETS.clear_move()
  84. if m != "BLIND":
  85. self.modes[m] = 0
  86. self.callback(m)
  87. out = 1
  88. return 1
  89. elif value:
  90. for m in self.modes:
  91. if m not in protected and mode not in protected and m != mode:
  92. if self.modes[m]:
  93. self.modes[m]= 0
  94. self.callback(m)
  95. if self.modes[mode]:
  96. if modes == "MOVE":
  97. PRESETS.clear_move()
  98. if modes == "COPY":
  99. PRESETS.clear_copy()
  100. self.modes[mode] = 0 # value
  101. else:
  102. self.modes[mode] = 1 #value
  103. out = 1
  104. else:
  105. self.modes[mode] = 0 #value
  106. if modes == "COPY":
  107. PRESETS.clear_copy()
  108. if modes == "MOVE":
  109. PRESETS.clear_move()
  110. self.callback(mode)
  111. return value
  112. def set_cb(self,cb):
  113. if cb:
  114. self.__cb = cb
  115. def callback(self,mode):
  116. if self.__cb is not None and mode in self.modes:
  117. value = self.modes[mode]
  118. self.__cb(mode=mode,value=value)
  119. modes = Modes()
  120. #modes.val("BLIND", 0)
  121. #modes.modes["BLIND"] = 0
  122. modes.modes["ESC"] = 0
  123. modes.modes["REC"] = 0
  124. modes.modes["EDIT"] = 0
  125. modes.modes["MOVE"] = 0
  126. modes.modes["FLASH"] = 0
  127. modes.modes["GO"] = 0
  128. modes.modes["DEL"] = 0
  129. modes.modes["STONY_FX"] = 0
  130. modes.modes["SELECT"] = 0
  131. modes.modes["CFG-BTN"] = 0
  132. modes.modes["LABEL"] = 0
  133. def xcb(mode,value=None):
  134. print("xcb","MODE CALLBACK",mode,value)
  135. if mode == "STONY_FX":
  136. print("xcb",modes.val("STONY_FX"))
  137. #modes.set_cb(xcb)
  138. POS = ["PAN","TILT","MOTION"]
  139. COLOR = ["RED","GREEN","BLUE","COLOR"]
  140. BEAM = ["GOBO","G-ROT","PRISMA","P-ROT","FOCUS","SPEED"]
  141. INT = ["DIM","SHUTTER","STROBE","FUNC"]
  142. #client = chat.tcp_sender(port=50001)
  143. jclient = chat.tcp_sender()#port=50001)
  144. def jclient_send(data):
  145. jclient.send("\00 "+ json.dumps(data) +"\00 ")
  146. class _FadeTime():
  147. def __init__(self):
  148. self._value = 2
  149. self._on = 1
  150. def inc(self,value=None):
  151. if value is not None:
  152. if type(value) is float:
  153. self._value += round(value,4)
  154. else:
  155. self._value += value
  156. return self._value
  157. def val(self,value=None):
  158. if value is not None:
  159. if type(value) is float:
  160. self._value = round(value,4)
  161. else:
  162. self._value = value
  163. return self._value
  164. def on(self):
  165. self._on = 1
  166. def off(self):
  167. self._on = 0
  168. def _is(self):
  169. if self._on:
  170. return 1
  171. return 0
  172. FADE = _FadeTime() #2 #0.1 #1.13
  173. fx_move_prm = {"SIZE":20,"SPEED":100,"OFFSET":50,"BASE":"-","START":0}
  174. fx_prm = {"SIZE":200,"SPEED":30,"OFFSET":255,"BASE":"-","START":0,"MODE":0,"MO":0,"DIR":1,"WING":2}
  175. fx_modes = [":RED",":GREEN",":BLUE",":MAG",":YELLOW",":CYAN"]
  176. fx_mo = ["sinus","on","on2","bump","bump2","fade","cosinus"]
  177. def build_cmd(dmx,val,args=[],flash=0,xpfx="",attr=""):
  178. if not args:
  179. args.append(FADE.val())
  180. cmd=""
  181. if xpfx:
  182. pfx=xpfx
  183. elif flash:
  184. pfx ="df"
  185. else:
  186. pfx ="d"
  187. if type(val) is float or type(val) is int:
  188. cmd += ",{}{}:{:0.4f}".format(pfx,dmx,val)
  189. else:
  190. cmd += ",{}{}:{}".format(pfx,dmx,val)
  191. if flash:
  192. cmd += ":0:0"#.format(val)
  193. else:
  194. for val in args:
  195. if type(val) is float or type(val) is int:
  196. cmd += ":{:0.4f}".format(val)
  197. else:
  198. cmd += ":{}".format(val)
  199. if attr:
  200. cmd += ":"+str(attr)
  201. cprint("build_cmd",cmd,color="red")
  202. return cmd
  203. def update_raw_dmx(data ,value=None,args=[],xfade=0,flash=0,pfx="d",fx=0):
  204. if flash:
  205. xfade = 0
  206. if not args: # and xfade is not None:# and FADE._is():
  207. args.append(xfade)
  208. else:
  209. args[0] = xfade
  210. cmd = []
  211. jcmd = []
  212. if flash:
  213. pfx += "f"
  214. for row in data:
  215. jxcmd={}
  216. if type(value) is float:
  217. jxcmd["VALUE"] = value #round(value,3)
  218. else:
  219. jxcmd["VALUE"] = value
  220. jxcmd["args"] = []
  221. if fx:
  222. if value is not None:
  223. # z.b. flush off
  224. xcmd = str(value)+":"+row["FX"].split(":",1)[-1]
  225. jxcmd["FX"] = row["FX"].split(":",1)[-1]
  226. else:
  227. xcmd = row["FX"]
  228. jxcmd["FX"] = row["FX"]
  229. if row["FX2"]:
  230. jxcmd["FX2"] = row["FX2"]
  231. else:
  232. if row["VALUE"] is None:
  233. xcmd = ""
  234. else:
  235. if value is not None:
  236. if type(value) is float:
  237. xcmd = "{:0.4f}".format(value)
  238. else:
  239. xcmd = "{}".format(value)
  240. else:
  241. v=row["VALUE"]
  242. xcmd = "{:0.4f}".format(v)
  243. cprint([v])
  244. if type(v) is float:
  245. jxcmd["VALUE"] = v #round(v,3)
  246. else:
  247. jxcmd["VALUE"] = v
  248. for arg in args:
  249. if type(arg) is float:
  250. xcmd += ":{:0.4f}".format(arg)
  251. jxcmd["args"].append(v)#round(arg,3))
  252. else:
  253. xcmd += ":{}".format(arg)
  254. jxcmd["args"].append(arg)#round(arg,3))
  255. #print( "pack: FIX",row["FIX"],row["ATTR"], xcmd)
  256. #xcmd += ":{}".format(row["ATTR"])
  257. v= xfade #FADE.val() #rxcmd["args"][0]
  258. if type( v ) is float:
  259. jxcmd["FADE"] = round(v,4)
  260. else:
  261. jxcmd["FADE"] = v
  262. #if ("VALUE" in jxcmd and jxcmd["VALUE"] is not None) or "FX" in jxcmd and jxcmd["FX"]:
  263. jcmd.append( jxcmd)
  264. cmd.append( xcmd)
  265. #if xcmd:
  266. # cprint("update_raw_dmx j",jxcmd,color="red")
  267. # cprint("update_raw_dmx x",xcmd,color="red")
  268. return cmd,jcmd
  269. def update_dmx(attr,data,value=None,args=None,flash=0,pfx=""):
  270. xfade = 0
  271. if not args:
  272. args=[]
  273. xfade = FADE.val()
  274. args.append(xfade)
  275. #global modes #BLIND
  276. #print("update_dmx",data)
  277. dmx = data["DMX"]
  278. dmx = (data["UNIVERS"]*512)+data["DMX"]
  279. val = None
  280. cmd=""
  281. try:
  282. if attr == "DIM" and data["ATTRIBUT"][attr]["NR"] < 0: #VDIM
  283. #print( "VDIM")
  284. for attr in data["ATTRIBUT"]:
  285. dmx = (data["UNIVERS"]*512) + data["DMX"]
  286. dmx = data["DMX"]
  287. if data["ATTRIBUT"][attr]["NR"] < 0: #virtual channels
  288. continue
  289. dmx += data["ATTRIBUT"][attr]["NR"]-1
  290. mode = ""
  291. if "MODE" in data["ATTRIBUT"][attr]:
  292. mode = data["ATTRIBUT"][attr]["MODE"]
  293. #print(attr)
  294. val = data["ATTRIBUT"][attr]["VALUE"]
  295. if data["ATTRIBUT"][attr]["MASTER"]:
  296. val = val * (data["ATTRIBUT"]["DIM"]["VALUE"] / 255.)
  297. if val is not None:
  298. #cmd += ",d{}:{:0.4f}".format(dmx,int(val))
  299. if value is not None:
  300. val = value
  301. if mode == "F": #FADE
  302. cmd += build_cmd(dmx,val,args=args,flash=flash,xpfx=pfx,attr=attr)
  303. else:
  304. cmd += build_cmd(dmx,val,args=[0],flash=flash,xpfx=pfx,attr=attr)
  305. #print("cmd",cmd)
  306. elif data["ATTRIBUT"][attr]["NR"] > 0:
  307. dmx += data["ATTRIBUT"][attr]["NR"]-1
  308. val = data["ATTRIBUT"][attr]["VALUE"]
  309. mode = ""
  310. if "MODE" in data["ATTRIBUT"][attr]:
  311. mode = data["ATTRIBUT"][attr]["MODE"]
  312. if data["ATTRIBUT"][attr]["MASTER"]:
  313. #if "VDIM" in data["ATTRIBUT"]:
  314. if "DIM" in data["ATTRIBUT"] and data["ATTRIBUT"]["DIM"]["NR"] < 0: #VDIM
  315. val = val * (data["ATTRIBUT"]["DIM"]["VALUE"] / 255.)
  316. if val is not None:
  317. #cmd += ",d{}:{}".format(dmx,int(val))
  318. if value is not None:
  319. val = value
  320. if mode == "F": #FADE
  321. cmd += build_cmd(dmx,val,args=args,flash=flash,xpfx=pfx,attr=attr)
  322. else:
  323. cmd += build_cmd(dmx,val,args=[0],flash=flash,xpfx=pfx,attr=attr)
  324. #print("cmd",cmd)
  325. if modes.val("BLIND"):
  326. cmd=""
  327. cprint("update_dmx",cmd,color="red")
  328. return cmd
  329. except Exception as e:
  330. cprint("== cb EXCEPT",e,color="red")
  331. cprint("Error on line {}".format(sys.exc_info()[-1].tb_lineno),color="red")
  332. cprint(''.join(traceback.format_exception(None, e, e.__traceback__)),color="red")
  333. raise e
  334. class dummy_event():
  335. def __init__(self):
  336. self.num =0
  337. self.type = 4 #press 5 release
  338. self.set_value=-1
  339. gcolor = 1
  340. def cprint(*text,color="blue",space=" ",end="\n"):
  341. #return 0 #disable print dbg
  342. if not gcolor:
  343. print(text)
  344. return 0
  345. if color == "green":
  346. txt = '\033[92m'
  347. elif color == "red":
  348. txt = '\033[0;31m\033[1m'
  349. elif color == "yellow":
  350. txt = '\033[93m\033[1m'
  351. elif color == "cyan":
  352. txt = '\033[96m'
  353. else:
  354. txt = '\033[94m'
  355. for t in text:
  356. txt += str(t ) +" "
  357. #HEADER = '\033[95m'
  358. #OKBLUE = '\033[94m'
  359. #OKCYAN = '\033[96m'
  360. #OKGREEN = '\033[92m'
  361. #WARNING = '\033[93m'
  362. #FAIL = '\033[91m'
  363. #ENDC = '\033[0m'
  364. #BOLD = '\033[1m'
  365. #UNDERLINE = '\033[4m'
  366. txt += '\033[0m'
  367. print(txt,end=end)
  368. #return txt
  369. cprint("________________________________")
  370. class Xevent():
  371. def __init__(self,fix,elem,attr=None,data=None,mode=None):
  372. self.fix = fix
  373. self.data=data
  374. self.attr = attr
  375. self.elem = elem
  376. self.mode = mode
  377. def fx(self,event):
  378. cprint("Xevent.fx",self.attr,self.fix,event)
  379. jdatas = []
  380. fx2 = {}
  381. if event.num == 4:
  382. cprint("FX:COLOR CHANGE",fx_prm,color="red")
  383. txt = "FX:RED"
  384. fx_prm["MODE"] += 1
  385. if fx_prm["MODE"] > len(fx_modes):
  386. fx_prm["MODE"]=0
  387. txt = "FX:"+fx_modes[fx_prm["MODE"]]
  388. master.elem_fx_commands["FX:RED"]["text"] = txt
  389. elif event.num == 5:
  390. cprint("FX:COLOR CHANGE",fx_prm,color="red")
  391. txt = "FX:RED"
  392. fx_prm["MODE"] -= 1
  393. if fx_prm["MODE"] < 0:
  394. fx_prm["MODE"]= len(fx_modes)-1
  395. txt = "FX:"+fx_modes[fx_prm["MODE"]]
  396. master.elem_fx_commands["FX:RED"]["text"] = txt
  397. elif event.num == 1:
  398. cmd = ""
  399. offset = 0
  400. offset_flag=0
  401. start = fx_prm["START"]
  402. base = fx_prm["BASE"]
  403. #FIXTURES.start_fx(attr)
  404. xfixtures = []
  405. # WING's and BLOCK's
  406. fix_active =FIXTURES.get_active()
  407. for fix in fix_active:
  408. if fix == "CFG":
  409. continue
  410. xfixtures.append(fix)
  411. x=0
  412. if fx_prm["DIR"] < 0:
  413. xfixtures = xfixtures[::-1]
  414. x=-1
  415. wings = []
  416. if fx_prm["WING"]:
  417. l = len(xfixtures)
  418. w = l // fx_prm["WING"]
  419. teiler = l//w
  420. if teiler < 2:
  421. teiler = 2
  422. for i in range(teiler):
  423. j = i*w
  424. wing = xfixtures[j:j+w]
  425. if i%2==0:
  426. wing = wing[::-1]
  427. print("wing",i,"j",j,"w",w,"wing",wing)
  428. wings.append(wing)
  429. if l > j+w:
  430. wing = xfixtures[j+w:]
  431. wings.append(wing)
  432. else:
  433. wings.append(xfixtures)
  434. print("FX442 ",xfixtures)
  435. print("FX442 ",fx_prm,x)
  436. for wing in wings:
  437. print("wing",wing)
  438. wlen = len(wing)
  439. coffset= 0 # 1024/wlen * (offset/255)
  440. offset = 0
  441. for fix in wing:
  442. data = FIXTURES.fixtures[fix]
  443. #print( "ADD FX",fix)
  444. for attr in data["ATTRIBUT"]:
  445. jdata = {"MODE":"FX"}
  446. jdata["VALUE"] = None
  447. jdata["FIX"] = fix
  448. jdata["DMX"] = FIXTURES.get_dmx(fix,attr)
  449. jdata["ATTR"] =attr
  450. if attr.endswith("-FINE"):
  451. continue
  452. csize = fx_prm["SIZE"]
  453. cspeed = fx_prm["SPEED"]
  454. cstart = fx_prm["START"]
  455. cbase = fx_prm["BASE"]
  456. #cstart = start
  457. coffset= round(offset,1)
  458. #cbase = base
  459. fx=""
  460. if "SIN" in self.attr:
  461. fx = "sinus"
  462. elif "FD" in self.attr:
  463. fx = "fade"
  464. elif "ON2" in self.attr:
  465. fx = "on2"
  466. elif "ON" in self.attr:
  467. fx = "on"
  468. elif "BUM2" in self.attr:
  469. fx = "bump2"
  470. elif "BUM" in self.attr:
  471. fx = "bump"
  472. elif "COS" in self.attr:
  473. fx = "cosinus"
  474. if fx:
  475. if fx_prm["SPEED"] < 0.1:
  476. fx = "off"
  477. else:
  478. if ":DIM" in self.attr:
  479. base=""
  480. ffxb= fx_mo[fx_prm["MO"]]
  481. if attr == "DIM":
  482. if fx_prm["SPEED"] < 0.1:
  483. fx = "off"
  484. else:
  485. fx = ffxb #"fade"
  486. elif ":TILT" in self.attr:
  487. base=""
  488. if attr == "PAN":
  489. fx = "off"
  490. if attr == "TILT":
  491. if fx_prm["SPEED"] < 0.1:
  492. fx = "off"
  493. else:
  494. fx = "sinus"
  495. elif ":PAN" in self.attr:
  496. base=""
  497. if attr == "PAN":
  498. if fx_prm["SPEED"] < 0.1:
  499. fx = "off"
  500. else:
  501. fx = "cosinus"
  502. if attr == "TILT":
  503. fx = "off"
  504. elif ":CIR" in self.attr:
  505. base=""
  506. if attr == "PAN":
  507. if fx_prm["SPEED"] < 0.1:
  508. fx = "off"
  509. else:
  510. fx = "cosinus"
  511. if attr == "TILT":
  512. if fx_prm["SPEED"] < 0.1:
  513. fx = "off"
  514. else:
  515. fx = "sinus"
  516. elif ":RED" in self.attr:
  517. ffxb= fx_mo[fx_prm["MO"]]
  518. ffx= "off" #fx_mo[fx_prm["MO"]]
  519. if ":RED" in fx_modes[fx_prm["MODE"]]:#
  520. base="-"
  521. if attr == "RED":
  522. #coffset=0
  523. #cspeed=0
  524. fx=ffx
  525. if attr == "GREEN":
  526. fx = ffxb# "off"
  527. if attr == "BLUE":
  528. fx = ffxb#"off"
  529. elif ":GREEN" in fx_modes[fx_prm["MODE"]]:#fx_prm["MODE"]:#in self.attr:
  530. base="-"
  531. if attr == "RED":
  532. fx = ffxb#"off"
  533. elif ":GREEN" in fx_modes[fx_prm["MODE"]]:#fx_prm["MODE"]:#in self.attr:
  534. if attr == "GREEN":
  535. fx = ffxb# "off"
  536. #cspeed=0
  537. #coffset=0
  538. fx=ffx
  539. if attr == "BLUE":
  540. fx = ffxb#"off"
  541. elif ":BLUE" in fx_modes[fx_prm["MODE"]]:#fx_prm["MODE"]:#self.attr:
  542. base="-"
  543. if attr == "RED":
  544. fx = ffxb# "off"
  545. if attr == "GREEN":
  546. fx = ffxb# "off"
  547. if attr == "BLUE":
  548. fx = ffxb# "off"
  549. #cspeed=0
  550. #coffset=0
  551. fx=ffx
  552. elif ":YELLOW" in fx_modes[fx_prm["MODE"]]:#fx_prm["MODE"]:#self.attr:
  553. base="-"
  554. if attr == "RED":
  555. fx = ffxb# "off"
  556. #cspeed=0
  557. #coffset=0
  558. fx=ffx
  559. if attr == "GREEN":
  560. fx = ffxb# "off"
  561. #cspeed=0
  562. #coffset=0
  563. fx=ffx
  564. if attr == "BLUE":
  565. fx = "off"
  566. elif ":CYAN" in fx_modes[fx_prm["MODE"]]:#fx_prm["MODE"]:#self.attr:
  567. base="-"
  568. if attr == "RED":
  569. fx = ffxb# "off"
  570. if attr == "GREEN":
  571. fx = ffxb# "off"
  572. #cspeed=0
  573. #coffset=0
  574. fx=ffx
  575. if attr == "BLUE":
  576. fx = ffxb# "off"
  577. #cspeed=0
  578. #coffset=0
  579. fx=ffx
  580. elif ":MAG" in fx_modes[fx_prm["MODE"]]:#fx_prm["MODE"]:#self.attr:
  581. base="-"
  582. if attr == "RED":
  583. fx = ffxb# "off"
  584. fx=ffx
  585. #cspeed=0
  586. #coffset=0
  587. if attr == "GREEN":
  588. fx = ffxb# "off"
  589. if attr == "BLUE":
  590. fx = ffxb# "off"
  591. fx=ffx
  592. #cspeed=0
  593. #coffset=0
  594. else:
  595. cprint("FX: unbekant",fx_modes[fx_prm["MODE"]],color="red")
  596. #if fx:
  597. fxtype = fx
  598. #if data["ATTRIBUT"][attr]["ACTIVE"] and fx:
  599. # fjdata = {}
  600. # fjdata["TYPE"] = fxtype
  601. # fjdata["SIZE"] = round(csize,2)
  602. # fjdata["SPEED"] = round(cspeed,2)
  603. # fjdata["START"] = cstart
  604. # fjdata["OFFSET"]= round(coffset,2)
  605. # fjdata["BASE"] = cbase
  606. # jdata["FX2"] = fjdata
  607. # print(jdata)
  608. # #jdatas.append(jdata)
  609. fxtype = fx
  610. if fx:
  611. #fx += ":{:0.0f}:{:0.0f}:{:0.0f}:{:0.0f}:{}:".format(fx_prm["SIZE"],fx_prm["SPEED"],start,offset,base)
  612. fx += ":{:0.0f}:{:0.0f}:{:0.0f}:{:0.0f}:{}:".format(csize,cspeed,cstart,coffset,cbase)
  613. #jdata["FX"] = fx
  614. offset_flag=1
  615. #print("ADD FX",fix,attr,fx,data["ATTRIBUT"][attr]["ACTIVE"])
  616. if "FX" not in data["ATTRIBUT"][attr]:
  617. data["ATTRIBUT"][attr]["FX"] =""
  618. if "FX2" not in data["ATTRIBUT"][attr]:
  619. data["ATTRIBUT"][attr]["FX2"] ={}
  620. if data["ATTRIBUT"][attr]["ACTIVE"] and fx:
  621. print("++ADD FX",fix,attr,fx)
  622. #data["ATTRIBUT"][attr]["FX"] = fx #"sinus:40:100:10"
  623. cmd+=update_dmx(attr,data,pfx="fx",value=fx)#,flash=FLASH)
  624. fjdata = {}
  625. fjdata["TYPE"] = fxtype
  626. fjdata["SIZE"] = round(csize,2)
  627. fjdata["SPEED"] = round(cspeed,2)
  628. fjdata["START"] = cstart
  629. fjdata["OFFSET"]= round(coffset,2)
  630. fjdata["BASE"] = cbase
  631. jdata["FX2"] = fjdata
  632. data["ATTRIBUT"][attr]["FX2"] = fjdata
  633. jdatas.append(jdata)
  634. print(jdata)
  635. if fx_prm["OFFSET"] > 0.5 and offset_flag:
  636. offset_flag=0
  637. #offset += fx_prm["OFFSET"] # add offset on next fixture
  638. #offset += 1024/wlen * (fx_prm["OFFSET"]/255)
  639. if fx_prm["DIR"]:
  640. offset += 1024/wlen * (fx_prm["OFFSET"]/255)
  641. else:
  642. offset -= 1024/wlen * (fx_prm["OFFSET"]/255)
  643. offset = round(offset,2)
  644. #print("offset",offset)
  645. if cmd and not modes.val("BLIND"):
  646. #client.send(cmd)
  647. jclient_send(jdatas)
  648. master.refresh_fix()
  649. def command(self,event):
  650. if self.mode == "COMMAND":
  651. if self.attr == "CLEAR":
  652. if event.num == 1:
  653. ok = FIXTURES.clear()
  654. if ok:
  655. master.refresh_fix()
  656. modes.val(self.attr,0)
  657. elif self.attr.startswith("SZ:"):#SIN":
  658. #global fx_prm
  659. k = "SIZE"
  660. if event.num == 1:
  661. pass
  662. elif event.num == 2:
  663. pass
  664. elif event.num == 4:
  665. if fx_prm[k] <= 0:
  666. fx_prm[k] = 1
  667. fx_prm[k] *=1.2
  668. elif event.num == 5:
  669. fx_prm[k] /=1.2
  670. #fx_prm[k] =int(fx_prm[k])
  671. if fx_prm[k] > 4000:
  672. fx_prm[k] = 4000
  673. if fx_prm[k] < 0:
  674. fx_prm[k] =0
  675. self.data.elem_fx_commands[self.attr]["text"] = "SZ:{:0.0f}".format(fx_prm[k])
  676. elif self.attr.startswith("SP:"):#SIN":
  677. #global fx_prm
  678. k = "SPEED"
  679. if event.num == 1:
  680. pass
  681. elif event.num == 2:
  682. pass
  683. elif event.num == 4:
  684. if fx_prm[k] <= 0:
  685. fx_prm[k] = 1
  686. fx_prm[k] *=1.2
  687. elif event.num == 5:
  688. fx_prm[k] /=1.2
  689. #fx_prm[k] =int(fx_prm[k])
  690. if fx_prm[k] > 4000:
  691. fx_prm[k] = 4000
  692. if fx_prm[k] < 0:
  693. fx_prm[k] =0
  694. if fx_prm[k] < 0.1:
  695. self.data.elem_fx_commands[self.attr]["text"] = "SP:off".format(fx_prm[k])
  696. else:
  697. self.data.elem_fx_commands[self.attr]["text"] = "SP:{:0.0f}".format(fx_prm[k])
  698. elif self.attr.startswith("ST:"):#SIN":
  699. #global fx_prm
  700. k = "START"
  701. if event.num == 1:
  702. pass
  703. elif event.num == 2:
  704. pass
  705. elif event.num == 4:
  706. if fx_prm[k] <= 0:
  707. fx_prm[k] = 1
  708. fx_prm[k] *=1.2
  709. elif event.num == 5:
  710. fx_prm[k] /=1.2
  711. #fx_prm[k] =int(fx_prm[k])
  712. if fx_prm[k] > 4000:
  713. fx_prm[k] = 4000
  714. if fx_prm[k] < 0:
  715. fx_prm[k] =0
  716. self.data.elem_fx_commands[self.attr]["text"] = "ST:{:0.0f}".format(fx_prm[k])
  717. elif self.attr.startswith("MO:"):# on,sinus,bump
  718. #global fx_prm
  719. k = "MO"
  720. if event.num == 1:
  721. pass
  722. elif event.num == 2:
  723. pass
  724. elif event.num == 4:
  725. fx_prm[k] -=1
  726. if fx_prm[k] < 0:
  727. fx_prm[k] = len(fx_mo)-1
  728. elif event.num == 5:
  729. fx_prm[k] +=1
  730. if fx_prm[k] >= len(fx_mo):
  731. fx_prm[k] = 0
  732. txt = fx_mo[fx_prm[k]]
  733. self.data.elem_fx_commands[self.attr]["text"] = "MO:{}".format(txt)
  734. elif self.attr.startswith("DIR:"):#SIN":
  735. #global fx_prm
  736. k = "DIR"
  737. if event.num == 1:
  738. pass
  739. elif event.num == 2:
  740. pass
  741. elif event.num == 4:
  742. fx_prm[k] = 1
  743. elif event.num == 5:
  744. fx_prm[k] =-1
  745. txt = fx_prm[k]
  746. self.data.elem_fx_commands[self.attr]["text"] = "DIR:{}".format(fx_prm[k])
  747. elif self.attr.startswith("WING:"):#SIN":
  748. #global fx_prm
  749. k = "WING"
  750. if event.num == 1:
  751. pass
  752. elif event.num == 2:
  753. pass
  754. elif event.num == 4:
  755. fx_prm[k] += 1
  756. elif event.num == 5:
  757. fx_prm[k] -=1
  758. if fx_prm[k] > 10:
  759. fx_prm[k] = 10
  760. if fx_prm[k] < 1:
  761. fx_prm[k] =1
  762. txt = fx_prm[k]
  763. self.data.elem_fx_commands[self.attr]["text"] = "WING:{}".format(fx_prm[k])
  764. elif self.attr.startswith("OF:"):#SIN":
  765. #global fx_prm
  766. k = "OFFSET"
  767. if event.num == 1:
  768. pass
  769. elif event.num == 2:
  770. pass
  771. elif event.num == 4:
  772. if fx_prm[k] <= 0:
  773. fx_prm[k] = 1
  774. fx_prm[k] *=1.2
  775. elif event.num == 5:
  776. fx_prm[k] /=1.2
  777. #fx_prm[k] =int(fx_prm[k])
  778. if fx_prm[k] > 255:
  779. fx_prm[k] = 255
  780. if fx_prm[k] < 0:
  781. fx_prm[k] =0
  782. self.data.elem_fx_commands[self.attr]["text"] = "OF:{:0.0f}".format(fx_prm[k])
  783. elif self.attr.startswith("BS:"):
  784. k = "BASE"
  785. if event.num == 1:
  786. fx_prm[k] = "0"
  787. elif event.num == 2:
  788. pass
  789. elif event.num == 4:
  790. fx_prm[k] = "+"
  791. elif event.num == 5:
  792. fx_prm[k] = "-"
  793. self.data.elem_fx_commands[self.attr]["text"] = "BS:{}".format(fx_prm[k])
  794. elif self.attr.startswith("FX:"):#SIN":
  795. self.fx(event)
  796. elif self.attr == "FX OFF":
  797. if event.num == 1:
  798. FIXTURES.fx_off("all")
  799. CONSOLE.fx_off("all")
  800. CONSOLE.flash_off("all")
  801. master.refresh_fix()
  802. return 0
  803. elif self.attr == "FADE":
  804. fade = FADE.val()
  805. print("EVENT CHANGE FADE",fade)
  806. if fade < 0.01:
  807. FADE.val(0.01)
  808. elif fade > 100.0:
  809. fade = 100
  810. if event.num == 4:
  811. fade *= 1.1
  812. elif event.num == 5:
  813. fade /= 1.1
  814. elif event.num == 1:
  815. if FADE._is():
  816. FADE.off()# = 0
  817. self.data.elem_commands[self.attr]["bg"] = "grey"
  818. else:
  819. FADE.on()# = 1
  820. self.data.elem_commands[self.attr]["bg"] = "green"
  821. elif event.num == 2:
  822. if fade > 1 and fade < 4:
  823. fade = 4
  824. elif fade > 3 and fade < 6:
  825. fade = 6
  826. elif fade > 5 and fade < 7:
  827. fade = 8
  828. elif fade > 7 and fade < 9:
  829. fade = 10
  830. elif fade > 9:
  831. fade = 0.01
  832. elif fade < 1:
  833. fade = 1.1
  834. fade = round(fade,3)
  835. FADE.val(fade)
  836. self.data.elem_commands[self.attr]["text"] = "Fade{:0.2f}".format(fade)
  837. elif self.attr == "BACKUP":
  838. modes.val(self.attr,1)
  839. PRESETS.backup_presets()
  840. FIXTURES.backup_patch()
  841. #time.sleep(1)
  842. modes.val(self.attr,0)
  843. else:
  844. if event.num == 1:
  845. print("ELSE",self.attr)
  846. modes.val(self.attr,1)
  847. return 0
  848. def cb(self,event):
  849. cprint("EVENT cb",self.attr,self.mode,event,color='yellow')
  850. print(["type",event.type,"num",event.num])
  851. try:
  852. change = 0
  853. if "keysym" in dir(event):
  854. if "Escape" == event.keysym:
  855. ok = FIXTURES.clear()
  856. master.refresh_fix()
  857. return 0
  858. if self.mode == "COMMAND":
  859. self.command(event)
  860. elif self.mode == "ROOT":
  861. if event.keysym=="Escape":
  862. pass
  863. elif self.mode == "INPUT":
  864. print("INP",self.data.entry.get())
  865. if event.keycode == 36:
  866. x=self.data.entry.get()
  867. #client.send(x)
  868. elif self.mode == "INPUT2":
  869. print("INP2",self.data.entry2.get())
  870. if event.keycode == 36:
  871. x=self.data.entry2.get()
  872. #client.send(x)
  873. elif self.mode == "INPUT3":
  874. print("INP3",self.data.entry3.get())
  875. if event.keycode == 36:
  876. x=self.data.entry3.get()
  877. #client.send(x)
  878. elif self.mode == "PRESET":
  879. nr = self.attr #int(self.attr.split(":")[1])-1
  880. if event.num == 1:
  881. if str(event.type) == '4': #4 ButtonPress
  882. if modes.val("REC"):
  883. self.data.preset_rec(nr)
  884. modes.val("REC",0)
  885. elif modes.val("DEL"):
  886. ok=PRESETS.delete(nr)
  887. if ok:
  888. modes.val("DEL",0)
  889. master.refresh_exec()
  890. elif modes.val("COPY"):
  891. ok=PRESETS.copy(nr)
  892. if ok:
  893. modes.val("COPY",0)
  894. master.refresh_exec()
  895. elif modes.val("MOVE"):
  896. ok=PRESETS.move(nr)
  897. if ok:
  898. modes.val("MOVE",0)
  899. master.refresh_exec()
  900. elif modes.val("CFG-BTN"):
  901. master.btn_cfg(nr)
  902. elif modes.val("LABEL"):#else:
  903. master.label(nr)
  904. elif modes.val("EDIT"):
  905. FIXTURES.clear()
  906. self.data.preset_select(nr)
  907. self.data.preset_go(nr,xfade=0,event=event,val=255)
  908. modes.val("EDIT", 0)
  909. master.refresh_fix()
  910. elif modes.val("SELECT"):
  911. self.data.preset_select(nr)
  912. else:
  913. self.data.preset_go(nr,event=event,val=255)
  914. else:
  915. self.data.preset_go(nr,xfade=0,event=event,val=0)
  916. if event.num == 3:
  917. if not modes.val("REC"):
  918. self.data.preset_go(nr,xfade=0,event=event,val=255)
  919. return 0
  920. elif self.mode == "INPUT":
  921. return 0
  922. if self.mode == "ENCODER":
  923. cprint("ENC",self.fix,self.attr,self.mode)
  924. cprint(self.data)
  925. val=""
  926. if event.num == 1:
  927. val ="click"
  928. elif event.num == 4:
  929. val ="+"
  930. elif event.num == 5:
  931. val ="-"
  932. if val:
  933. FIXTURES.encoder(fix=self.fix,attr=self.attr,xval=val)
  934. master.refresh_fix()
  935. except Exception as e:
  936. cprint("== cb EXCEPT",e,color="red")
  937. cprint("Error on line {}".format(sys.exc_info()[-1].tb_lineno),color="red")
  938. cprint(''.join(traceback.format_exception(None, e, e.__traceback__)),color="red")
  939. def wheel(event,d=None):
  940. print("wheel",event,d)
  941. import copy
  942. class Element():
  943. def __init__(self):
  944. self.__data = {}
  945. def set(self,key,val):
  946. self.__data[key] = val
  947. class Base():
  948. def __init__(self):
  949. show_name = "GloryCamp2021"
  950. #show_name = "JMS"
  951. #show_name = "Dimmer"
  952. show_name = "DemoShow"
  953. self.home = os.environ['HOME']
  954. self.show_path = self.home +"/LibreLight/"
  955. if not os.path.isdir(self.show_path):
  956. os.mkdir(self.show_path)
  957. self.show_path += "/show/"
  958. if not os.path.isdir(self.show_path):
  959. os.mkdir(self.show_path)
  960. self.show_path += "/" +show_name +"/"
  961. if not os.path.isdir(self.show_path):
  962. os.mkdir(self.show_path)
  963. pass
  964. def _load(self,filename):
  965. xfname = self.show_path+"/"+str(filename)+".sav"
  966. print("load",xfname)
  967. f = open(xfname,"r")
  968. lines = f.readlines()
  969. f.close()
  970. data = OrderedDict()
  971. labels = OrderedDict()
  972. for line in lines:
  973. key,label,rdata = line.split("\t",2)
  974. key = int(key)
  975. #print(xfname,"load",key,label)
  976. #print(line)
  977. jdata = json.loads(rdata,object_pairs_hook=OrderedDict)
  978. nrnull = 0
  979. print(jdata)
  980. #if "ATTRIBUT" in jdata: # translate old FIXTURES.fixtures start with 0 to 1
  981. # for attr in jdata["ATTRIBUT"]:
  982. # row = jdata["ATTRIBUT"][attr]
  983. # if type(row) is OrderedDict:
  984. # #print(row)
  985. # if "VALUE" in row:
  986. # v = row["VALUE"]
  987. # if type(v) is float:
  988. # v = round(v,4)
  989. # jdata["ATTRIBUT"][attr]["VALUE"] = round(v,4)
  990. # print("preset v",key,label,attr,v)
  991. if "ATTRIBUT" in jdata: # translate old FIXTURES.fixtures start with 0 to 1
  992. for attr in jdata["ATTRIBUT"]:
  993. pass
  994. #if "VALUE" in jdata["ATTRIBUT"][attr]:
  995. # v = jdata["ATTRIBUT"][attr]["VALUE"]
  996. # if type(v) is float:
  997. # jdata["ATTRIBUT"][attr]["VALUE"] = round(v,4)
  998. # #print("fix v",attr,v)
  999. #if "NR" in jdata["ATTRIBUT"][attr]:
  1000. # nr = jdata["ATTRIBUT"][attr]["NR"]
  1001. # if nr == 0:
  1002. # nrnull = 1
  1003. # break
  1004. if nrnull:
  1005. print("DMX NR IS NULL",attr,"CHANGE +1")
  1006. for attr in jdata["ATTRIBUT"]:
  1007. if "NR" in jdata["ATTRIBUT"][attr]:
  1008. nr = jdata["ATTRIBUT"][attr]["NR"]
  1009. if nr >= 0:
  1010. jdata["ATTRIBUT"][attr]["NR"] +=1
  1011. data[key] = jdata
  1012. labels[key] = label
  1013. return data,labels
  1014. def _backup(self,filename,data,labels):
  1015. #fixture
  1016. #xfname = "show/"+show_name+"/"+str(filename)+".sav"
  1017. xfname = self.show_path+"/"+str(filename)+".sav"
  1018. print("backup",xfname)
  1019. f = open(xfname,"w")
  1020. for key in data:
  1021. line = data[key]
  1022. #print(line)
  1023. label = "label"
  1024. if key in labels:
  1025. label = labels[key]
  1026. if label == "Name-"+str(key):
  1027. label = ""
  1028. #print(xfname,"load",key,label,len(line))
  1029. f.write( "{}\t{}\t{}\n".format( key,label,json.dumps(line) ) )
  1030. f.close()
  1031. class Event():
  1032. def __init__(self,name):
  1033. self.name=name
  1034. #print("init",self)
  1035. def event(self,event):
  1036. print(self.name,event)
  1037. class scroll():
  1038. def __init__(self,canvas):
  1039. self.canvas=canvas
  1040. def config(self,event):
  1041. canvas = self.canvas
  1042. canvas.configure(scrollregion=canvas.bbox("all"))#,width=400,height=200)
  1043. def hex_to_rgb(hex):
  1044. return tuple(int(hex[i:i+2], 16) for i in (0, 2, 4))
  1045. class cb():
  1046. def __init__(self,win):
  1047. self.win = win
  1048. def _callback(self,event):
  1049. clobj=event.widget
  1050. ## undermouse=find_withtag(master.CURRENT)
  1051. undermouse=self.win.find_closest(self.win.CURRENT)
  1052. print( repr(undermouse))
  1053. def callback(self,event):
  1054. print(__file__,self,"callback",event)
  1055. cnv = self.win
  1056. item = cnv.find_closest(cnv.canvasx(event.x), cnv.canvasy(event.y))[0]
  1057. tags = cnv.gettags(item)
  1058. #cnv.itemconfigure(self.tag, text=tags[0])
  1059. print(tags,item)
  1060. color = cnv.itemcget(item, "fill")
  1061. cnv.itemconfig("all", width=1)#filla="green")
  1062. cnv.itemconfig(item, width=3)#filla="green")
  1063. print(color)
  1064. print( hex_to_rgb(color[1:]))
  1065. class GUI(Base):
  1066. def __init__(self):
  1067. super().__init__()
  1068. self.load()
  1069. self._XX = 0
  1070. self.all_attr =["DIM","PAN","TILT"]
  1071. self.elem_attr = {}
  1072. self.fx_commands =["STONY_FX","FX OFF","\n"
  1073. ,"FX:CIR","FX:PAN","FX:TILT","\n"
  1074. ,"MSZ:","MSP:","MST:","MOF:","MBS:-","\n"
  1075. ,"FX:DIM","FX:RED", "MO:on","DIR:1","WING:2","\n"
  1076. ,"SZ:","SP:","ST:","OF:","BS:-","\n"
  1077. , "FX:SIN","FX:COS","FX:BUM","FX:BUM2","FX:FD","FX:ON","FX:ON2" ]
  1078. self.commands =["\n","ESC","CFG-BTN","LABEL","BACKUP","DEL","\n"
  1079. ,"SELECT","FLASH","GO","FADE","MOVE","\n"
  1080. ,"BLIND","CLEAR","REC","EDIT","COPY","\n"
  1081. ]
  1082. self.elem_fx_commands = {}
  1083. self.val_fx_commands = {}
  1084. self.elem_commands = {}
  1085. self.val_commands = {}
  1086. self.elem_presets = {}
  1087. for i in range(8*8*8):
  1088. if i not in PRESETS.val_presets:
  1089. name = "Preset:"+str(i+1)+":\nXYZ"
  1090. #self.presets[i] = [i]
  1091. PRESETS.val_presets[i] = OrderedDict() # FIX
  1092. PRESETS.val_presets[i]["CFG"] = OrderedDict() # CONFIG
  1093. PRESETS.label_presets[i] = "-"
  1094. modes.set_cb(self.xcb)
  1095. def button_refresh(self,name,color,color2=None,fg=None):
  1096. cprint("button_refresh",name,color)
  1097. #if color == "gold":
  1098. # color2 = "yellow"
  1099. if color2 is None:
  1100. color2 = color
  1101. if name in self.elem_commands:
  1102. self.elem_commands[name]["bg"] = color
  1103. self.elem_commands[name].config(activebackground=color2)
  1104. if fg:
  1105. self.elem_commands[name]["fg"] = fg
  1106. print(dir(self.elem_commands[name]))
  1107. elif name in self.elem_fx_commands:
  1108. #todo
  1109. self.elem_fx_commands[name]["bg"] = color
  1110. self.elem_fx_commands[name].config(activebackground=color2)
  1111. if fg:
  1112. self.elem_fx_commands[name]["fg"] = fg
  1113. print(dir(self.elem_fx_commands[name]))
  1114. def btn_cfg(self,nr):
  1115. txt = PRESETS.btn_cfg(nr)
  1116. txt = tkinter.simpledialog.askstring("CFG-BTN","GO=GO\nFL=FLASH\nSEL=SELECT\n EXE:"+str(nr+1),initialvalue=txt)
  1117. if txt:
  1118. PRESETS.btn_cfg(nr,txt)
  1119. self.elem_presets[nr]["text"] = PRESETS.get_btn_txt(nr)
  1120. modes.val("CFG-BTN",0)
  1121. def label(self,nr):
  1122. txt = PRESETS.label(nr)
  1123. txt = tkinter.simpledialog.askstring("LABEL","EXE:"+str(nr+1),initialvalue=txt)
  1124. if txt:
  1125. PRESETS.label(nr,txt)
  1126. self.elem_presets[nr]["text"] = PRESETS.get_btn_txt(nr)
  1127. modes.val("LABEL", 0)
  1128. def xcb(self,mode,value=None):
  1129. cprint("MODE CALLBACK",mode,value,color="green",end="")
  1130. #cprint(self,"xcb","MODE CALLBACK",mode,value,color="green")
  1131. if value:
  1132. cprint("===== ON ======",color="red")
  1133. self.button_refresh(mode,color="red")#,fg="blue")
  1134. else:
  1135. cprint("===== OFF ======",color="red")
  1136. self.button_refresh(mode,color="lightgrey")#,fg="black")
  1137. def load(self,fname=""):
  1138. pass
  1139. def exit(self):
  1140. print("__del__",self)
  1141. PRESETS.backup_presets()
  1142. #print("********************************************************")
  1143. FIXTURES.backup_patch()
  1144. #print("*********del",self,"***********************************************")
  1145. def refresh_exec(self):
  1146. cprint("PRESET.refresh_exec()")
  1147. self._XX +=1
  1148. for k in PRESETS.val_presets:
  1149. label = ""
  1150. if k not in self.elem_presets:
  1151. cprint("ERROR",k ,"not in elem_presets continue")
  1152. continue
  1153. if k in PRESETS.label_presets:
  1154. label = PRESETS.label_presets[k]
  1155. #print([label])
  1156. b = self.elem_presets[k]
  1157. if k in PRESETS.val_presets and len(PRESETS.val_presets[k]) :
  1158. sdata = PRESETS.val_presets[k]
  1159. #print("sdata7654",sdata)
  1160. BTN="go"
  1161. if "CFG" in sdata:#["BUTTON"] = "GO"
  1162. if "BUTTON" in sdata["CFG"]:
  1163. BTN = sdata["CFG"]["BUTTON"]
  1164. txt=str(k)+":"+str(BTN)+":"+str(len(sdata)-1)+"\n"+label
  1165. #txt+=str(self._XX)
  1166. b["text"] = txt
  1167. b["bg"] = "yellow"
  1168. b.config(activebackground="yellow")
  1169. if len(sdata) > 1:
  1170. fx_color = 0
  1171. val_color = 0
  1172. for fix in sdata:
  1173. if fix == "CFG":
  1174. continue
  1175. #print( "$$$$",fix,sdata[fix])
  1176. for attr in sdata[fix]:
  1177. if "FX2" in sdata[fix][attr]:
  1178. if sdata[fix][attr]["FX2"]:
  1179. fx_color = 1
  1180. if "FX" in sdata[fix][attr]:
  1181. if sdata[fix][attr]["FX"]:
  1182. fx_color = 1
  1183. if "VALUE" in sdata[fix][attr]:
  1184. if sdata[fix][attr]["VALUE"] is not None:
  1185. val_color = 1
  1186. b["fg"] = "black"
  1187. if val_color:
  1188. b["bg"] = "gold"
  1189. b.config(activebackground="#ffaa55")
  1190. if fx_color:
  1191. b["fg"] = "blue"
  1192. else:
  1193. if fx_color:
  1194. b["bg"] = "cyan"
  1195. b.config(activebackground="#55d4ff")
  1196. else:
  1197. b["bg"] = "grey"
  1198. b.config(activebackground="#aaaaaa")
  1199. if "\n" in txt:
  1200. txt = txt.split("\n")[0]
  1201. if "SEL" in txt:
  1202. b["fg"] = "black"
  1203. b["bg"] = "#5555ff"
  1204. b.config(activebackground="#6666ff")
  1205. elif "GO" in txt:
  1206. b["fg"] = "black"
  1207. elif "FL" in txt:
  1208. b["fg"] = "#7f00ff"
  1209. def refresh_fix(self):
  1210. for fix in FIXTURES.fixtures:
  1211. sdata = FIXTURES.fixtures[fix]
  1212. for attr in sdata["ATTRIBUT"]:
  1213. if "FINE" in attr:
  1214. continue
  1215. v2 = sdata["ATTRIBUT"][attr]["VALUE"]
  1216. if fix in self.elem_attr:
  1217. elem = self.elem_attr[fix][attr]
  1218. #print( attr,v2)
  1219. elem["text"] = "{} {:0.2f}".format(attr,v2)
  1220. if sdata["ATTRIBUT"][attr]["ACTIVE"]:
  1221. elem["bg"] = "yellow"
  1222. elem.config(activebackground="yellow")
  1223. else:
  1224. elem["bg"] = "grey"
  1225. elem.config(activebackground="grey")
  1226. if sdata["ATTRIBUT"][attr]["FX"]:
  1227. elem["fg"] = "blue"
  1228. elif sdata["ATTRIBUT"][attr]["FX2"]:
  1229. elem["fg"] = "red"
  1230. else:
  1231. elem["fg"] = "black"
  1232. def preset_rec(self,nr):
  1233. print("------- STORE PRESET")
  1234. data = FIXTURES.get_active()
  1235. if modes.val("STONY_FX"):
  1236. PRESETS.rec(nr,data,"STONY_FX")
  1237. modes.val("STONY_FX",0)
  1238. else:
  1239. PRESETS.rec(nr,data)
  1240. sdata=data
  1241. PRESETS.val_presets[nr] = sdata
  1242. master.refresh_exec()
  1243. return 1
  1244. def preset_select(self,nr):
  1245. print("SELECT PRESET")
  1246. sdata = PRESETS.val_presets[nr]
  1247. cmd = ""
  1248. for fix in sdata:
  1249. if fix == "CFG":
  1250. continue
  1251. for attr in sdata[fix]:
  1252. v2 = sdata[fix][attr]["VALUE"]
  1253. v2_fx = sdata[fix][attr]["FX"]
  1254. #print( self.data.elem_attr)
  1255. if fix in self.elem_attr:
  1256. elem = self.elem_attr[fix][attr]
  1257. FIXTURES.fixtures[fix]["ATTRIBUT"][attr]["ACTIVE"] = 1
  1258. elem["bg"] = "yellow"
  1259. def preset_go(self,nr,val=None,xfade=None,event=None):
  1260. if xfade is None and FADE._is():
  1261. xfade = FADE.val()
  1262. print("GO PRESET FADE",nr,val)
  1263. rdata = PRESETS.get_raw_map(nr)
  1264. if not rdata:
  1265. return 0
  1266. print("???????")
  1267. cfg = PRESETS.get_cfg(nr)
  1268. print("''''''''")
  1269. #virtcmd = FIXTURES.get_virtual(rdata)
  1270. if not cfg:
  1271. cprint("NO CFG",cfg,nr)
  1272. return 0
  1273. xFLASH = 0
  1274. value=None
  1275. cprint("preset_go",nr,cfg)
  1276. if modes.val("SELECT") or ( "BUTTON" in cfg and cfg["BUTTON"] == "SEL") and val: #FLASH
  1277. self.preset_select(nr)
  1278. elif modes.val("FLASH") or ( "BUTTON" in cfg and cfg["BUTTON"] == "FL"): #FLASH
  1279. xFLASH = 1
  1280. xfade = 0
  1281. if type(val) is not None and val == 0 :
  1282. value = "off"
  1283. if event:
  1284. if str(event.type) == "ButtonRelease" or event.type == '5' :
  1285. # 4 fix vor ThinkPad / Debian 11
  1286. if xFLASH:
  1287. value = "off"
  1288. cprint("preset_go() FLUSH",value,color="red")
  1289. fcmd = FIXTURES.update_raw(rdata,update=0)
  1290. self._preset_go(rdata,cfg,fcmd,value,xfade=xfade,xFLASH=xFLASH)
  1291. elif not val:
  1292. cprint("preset_go() STOP",value,color="red")
  1293. elif modes.val("GO") or ( "BUTTON" in cfg and cfg["BUTTON"] in ["go","GO"]):
  1294. fcmd = FIXTURES.update_raw(rdata)
  1295. self._preset_go(rdata,cfg,fcmd,value,xfade=xfade,xFLASH=xFLASH)
  1296. if not (modes.val("FLASH") or ( "BUTTON" in cfg and cfg["BUTTON"] == "FL")): #FLASH
  1297. self.refresh_exec()
  1298. self.refresh_fix()
  1299. def _preset_go(self,rdata,cfg,fcmd,value,xfade=None,event=None,xFLASH=0):
  1300. if xfade is None and FADE._is():
  1301. xfade = FADE.val()
  1302. cprint("PRESETS._preset_go()",len(rdata))
  1303. vvcmd,jvvcmd = update_raw_dmx( rdata ,value,[],xfade=xfade )
  1304. fxcmd,jfxcmd = update_raw_dmx( rdata ,value,[],xfade=xfade,fx=1)
  1305. cmd = []
  1306. for vcmd,d in [[jvvcmd,"d"],[jfxcmd,"fx"]]:
  1307. if xFLASH:
  1308. d+="f"
  1309. for i,v in enumerate(fcmd):
  1310. if xFLASH:
  1311. vcmd[i]["FLASH"] = 1
  1312. DMX = fcmd[i]["DMX"]
  1313. if "VALUE" in vcmd[i] and type(vcmd[i]["VALUE"]) is float:
  1314. vcmd[i]["VALUE"] = round(vcmd[i]["VALUE"],3)
  1315. if DMX and vcmd[i]:
  1316. vcmd[i]["DMX"] = DMX
  1317. if "VIRTUAL" in fcmd[i]:
  1318. for a in fcmd[i]["VIRTUAL"]:
  1319. DMX = fcmd[i]["VIRTUAL"][a]
  1320. if DMX and vcmd[i]:
  1321. vcmd[i]["DMX"] = DMX
  1322. #if vcmd[i]["VALUE"] is not None or ("FX2" in vcmd[i] and vcmd[i]["FX2"]):
  1323. # cprint("jvcmd",vcmd[i])
  1324. # cmd.append(vcmd[i])
  1325. #elif vcmd[i]["VALUE"] is not None or ("FX" in vcmd[i] and vcmd[i]["FX"]):
  1326. # cprint("jvcmd",vcmd[i])
  1327. # cmd.append(vcmd[i])
  1328. cprint("jvcmd",vcmd[i])
  1329. cmd.append(vcmd[i])
  1330. if cmd and not modes.val("BLIND"):
  1331. jclient_send(cmd)
  1332. def draw_sub_dim(self,fix,data,c=0,r=0,frame=None):
  1333. i=0
  1334. if frame is None:
  1335. frame = tk.Frame(root,bg="black")
  1336. frame.pack(fill=tk.X, side=tk.TOP)
  1337. if fix not in self.elem_attr:
  1338. self.elem_attr[fix] = {}
  1339. for attr in data["ATTRIBUT"]:
  1340. if attr not in self.all_attr:
  1341. self.all_attr.append(attr)
  1342. if attr not in self.elem_attr[fix]:
  1343. self.elem_attr[fix][attr] = []
  1344. if attr.endswith("-FINE"):
  1345. continue
  1346. v= data["ATTRIBUT"][attr]["VALUE"]
  1347. b = tk.Button(frame,bg="lightblue", text=""+str(fix)+" "+data["NAME"],width=4)
  1348. b.bind("<Button>",Xevent(fix=fix,mode="D-SELECT",elem=b).cb)
  1349. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1350. c+=1
  1351. b = tk.Button(frame,bg="grey", text=str(attr)+' '+str(round(v,2)),width=6)
  1352. self.elem_attr[fix][attr] = b
  1353. b.bind("<Button>",Xevent(fix=fix,elem=b,attr=attr,mode="ENCODER",data=data).cb)
  1354. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1355. c+=1
  1356. if c >=12:
  1357. c=0
  1358. r+=1
  1359. return c,r
  1360. def draw_patch(self,yframe):
  1361. xframe = tk.Frame(yframe,bg="black")
  1362. xframe.pack()
  1363. def yview(event):
  1364. print("yevent",event)
  1365. yyy=20.1
  1366. xframe.yview_moveto(yyy)
  1367. i=0
  1368. c=0
  1369. r=0
  1370. for fix in FIXTURES.fixtures:
  1371. i+=1
  1372. data = FIXTURES.fixtures[fix]
  1373. b = tk.Button(xframe,bg="lightblue", text="FIX:"+str(fix)+" "+data["NAME"],width=20)
  1374. #b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  1375. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1376. c+=1
  1377. #r+=1
  1378. if fix not in self.elem_attr:
  1379. self.elem_attr[fix] = {}
  1380. patch = ["DMX","UNIVERS"]
  1381. for k in patch:
  1382. v=data[k]
  1383. b = tk.Button(xframe,bg="grey", text=str(k)+' '+str(v),width=8)
  1384. #self.elem_attr[fix][attr] = b
  1385. #b.bind("<Button>",Xevent(fix=fix,elem=b,attr=attr,data=data).cb)
  1386. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1387. c+=1
  1388. if c >=8:
  1389. c=1
  1390. r+=1
  1391. for attr in data["ATTRIBUT"]:
  1392. if attr not in self.all_attr:
  1393. self.all_attr.append(attr)
  1394. if attr not in self.elem_attr[fix]:
  1395. self.elem_attr[fix][attr] = []
  1396. if attr.endswith("-FINE"):
  1397. continue
  1398. v= data["ATTRIBUT"][attr]["VALUE"]
  1399. b = tk.Button(xframe,bg="grey", text=str(attr)+' '+str(round(v,2)),width=8)
  1400. #self.elem_attr[fix][attr] = b
  1401. #b.bind("<Button>",Xevent(fix=fix,elem=b,attr=attr,data=data).cb)
  1402. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1403. c+=1
  1404. if c >=8:
  1405. c=1
  1406. r+=1
  1407. c=0
  1408. r+=1
  1409. def draw_fix(self,xframe,yframe=None):
  1410. r=0
  1411. c=0
  1412. frame_dim=xframe
  1413. if yframe:
  1414. frame_dim=yframe
  1415. frame_fix=xframe
  1416. root = frame_dim
  1417. dim_frame = tk.Frame(root,bg="black")
  1418. dim_frame.pack(fill=tk.X, side=tk.TOP)
  1419. root = frame_fix
  1420. fix_frame = tk.Frame(root,bg="black")
  1421. fix_frame.pack(fill=tk.X, side=tk.TOP)
  1422. i=0
  1423. c=0
  1424. r=0
  1425. dim_end=0
  1426. for fix in FIXTURES.fixtures:
  1427. i+=1
  1428. data = FIXTURES.fixtures[fix]
  1429. #print("draw_fix", fix ,data )
  1430. if(len(data["ATTRIBUT"].keys()) <= 1):
  1431. c,r=self.draw_sub_dim(fix,data,c=c,r=r,frame=dim_frame)
  1432. else:
  1433. if not dim_end:
  1434. dim_end=1
  1435. c=0
  1436. r=0
  1437. #self._draw_fix(fix,data,root=fix_frame)
  1438. frame = fix_frame
  1439. b = tk.Button(frame,bg="lightblue", text="FIX:"+str(fix)+" "+data["NAME"],width=20)
  1440. b.bind("<Button>",Xevent(fix=fix,mode="SELECT",elem=b).cb)
  1441. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1442. c+=1
  1443. #r+=1
  1444. if fix not in self.elem_attr:
  1445. self.elem_attr[fix] = {}
  1446. for attr in data["ATTRIBUT"]:
  1447. if attr.endswith("-FINE"):
  1448. continue
  1449. if attr not in self.all_attr:
  1450. self.all_attr.append(attr)
  1451. if attr not in self.elem_attr[fix]:
  1452. self.elem_attr[fix][attr] = ["line1348",fix,attr]
  1453. v= data["ATTRIBUT"][attr]["VALUE"]
  1454. b = tk.Button(frame,bg="grey", text=str(attr)+' '+str(round(v,2)),width=8)
  1455. self.elem_attr[fix][attr] = b
  1456. b.bind("<Button>",Xevent(fix=fix,elem=b,attr=attr,mode="ENCODER",data=data).cb)
  1457. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1458. c+=1
  1459. if c >=8:
  1460. c=1
  1461. r+=1
  1462. c=0
  1463. r+=1
  1464. def draw_enc(self,xframe):
  1465. root2 = xframe
  1466. i=0
  1467. c=0
  1468. r=0
  1469. frame = tk.Frame(root2,bg="black")
  1470. frame.pack( side=tk.TOP,expand=1,fill="both")
  1471. b = tk.Button(frame,bg="lightblue", text="ENCODER",width=6)
  1472. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1473. c+=1
  1474. for attr in self.all_attr:
  1475. if attr.endswith("-FINE"):
  1476. continue
  1477. v=0
  1478. b = tk.Button(frame,bg="orange", text=str(attr)+'',width=6)
  1479. b.bind("<Button>",Xevent(fix=0,elem=b,attr=attr,data=self,mode="ENCODER").cb)
  1480. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1481. c+=1
  1482. if c >=8:
  1483. c=0
  1484. r+=1
  1485. def draw_fx(self,xframe):
  1486. frame_fx=xframe
  1487. i=0
  1488. c=0
  1489. r=0
  1490. frame = tk.Frame(frame_fx,bg="black")
  1491. frame.pack(fill=tk.X, side=tk.TOP)
  1492. b = tk.Button(frame,bg="lightblue", text="FX.",width=6)
  1493. #b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  1494. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1495. c+=1
  1496. for comm in self.fx_commands:
  1497. if comm == "\n":
  1498. c=0
  1499. r+=1
  1500. continue
  1501. v=0
  1502. b = tk.Button(frame,bg="lightgrey", text=str(comm),width=6,height=2)
  1503. if comm not in self.elem_fx_commands:
  1504. self.elem_fx_commands[comm] = b
  1505. self.val_fx_commands[comm] = 0
  1506. b.bind("<Button>",Xevent(fix=0,elem=b,attr=comm,data=self,mode="COMMAND").cb)
  1507. if comm == "BLIND":
  1508. b["bg"] = "grey"
  1509. elif comm == "CLEAR":
  1510. b["bg"] = "grey"
  1511. elif comm == "STONY_FX":
  1512. b["bg"] = "grey"
  1513. elif comm == "FADE":
  1514. b["bg"] = "green"
  1515. elif comm == "FX OFF":
  1516. b["bg"] = "magenta"
  1517. elif comm[:3] == "FX:":
  1518. b["text"] = comm #"BS:{}".format(fx_prm["BASE"])
  1519. b["bg"] = "#ffbf00"
  1520. elif comm == "MO:on":
  1521. b["text"] = comm #"BS:{}".format(fx_prm["BASE"])
  1522. b["bg"] = "lightgreen"
  1523. elif comm == "MO:on":
  1524. b["text"] = comm #"BS:{}".format(fx_prm["BASE"])
  1525. b["bg"] = "lightgreen"
  1526. elif comm == "SZ:":
  1527. b["text"] = "SZ:{:0.0f}".format(fx_prm["SIZE"])
  1528. b["bg"] = "lightgreen"
  1529. elif comm == "SP:":
  1530. b["text"] = "SP:{:0.0f}".format(fx_prm["SPEED"])
  1531. b["bg"] = "lightgreen"
  1532. elif comm == "ST:":
  1533. b["bg"] = "lightgreen"
  1534. b["text"] = "ST:{:0.0f}".format(fx_prm["START"])
  1535. elif comm == "OF:":
  1536. b["bg"] = "lightgreen"
  1537. b["text"] = "OF:{:0.0f}".format(fx_prm["OFFSET"])
  1538. elif comm == "BS:-":
  1539. b["bg"] = "lightgreen"
  1540. b["text"] = "BS:{}".format(fx_prm["BASE"])
  1541. elif comm[0] == "M":
  1542. b["text"] = comm #"BS:{}".format(fx_prm["BASE"])
  1543. b["bg"] = "lightgrey"
  1544. if comm:
  1545. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1546. c+=1
  1547. if c >=5:
  1548. c=0
  1549. r+=1
  1550. def draw_command(self,xframe):
  1551. frame_cmd=xframe
  1552. i=0
  1553. c=0
  1554. r=0
  1555. frame = tk.Frame(frame_cmd,bg="black")
  1556. frame.pack(fill=tk.X, side=tk.TOP)
  1557. b = tk.Button(frame,bg="lightblue", text="COMM.",width=6)
  1558. #b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  1559. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1560. #r+=1
  1561. c+=1
  1562. for comm in self.commands:
  1563. if comm == "\n":
  1564. c=0
  1565. r+=1
  1566. continue
  1567. v=0
  1568. b = tk.Button(frame,bg="lightgrey", text=str(comm),width=6,height=2)
  1569. if comm not in self.elem_commands:
  1570. self.elem_commands[comm] = b
  1571. self.val_commands[comm] = 0
  1572. b.bind("<Button>",Xevent(fix=0,elem=b,attr=comm,data=self,mode="COMMAND").cb)
  1573. if comm == "BLIND":
  1574. b["bg"] = "grey"
  1575. if comm == "CLEAR":
  1576. b["bg"] = "grey"
  1577. if comm == "STONY_FX":
  1578. b["bg"] = "grey"
  1579. if comm == "FADE":
  1580. b["bg"] = "green"
  1581. if comm == "FX OFF":
  1582. b["bg"] = "magenta"
  1583. if comm == "SZ:":
  1584. b["text"] = "SZ:{:0.0f}".format(fx_prm["SIZE"])
  1585. if comm == "SP:":
  1586. b["text"] = "SP:{:0.0f}".format(fx_prm["SPEED"])
  1587. if comm == "FADE":
  1588. b["text"] = "FADE:{:0.02f}".format(FADE.val())
  1589. if comm == "ST:":
  1590. b["text"] = "ST:{:0.0f}".format(fx_prm["START"])
  1591. if comm == "OF:":
  1592. b["text"] = "OF:{:0.0f}".format(fx_prm["OFFSET"])
  1593. if comm == "BS:":
  1594. b["text"] = "BS:{}".format(fx_prm["BASE"])
  1595. if comm:
  1596. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1597. c+=1
  1598. if c >=5:
  1599. c=0
  1600. r+=1
  1601. def draw_preset(self,xframe):
  1602. i=0
  1603. c=0
  1604. r=0
  1605. root = xframe
  1606. frame = tk.Frame(root,bg="black")
  1607. frame.pack(fill=tk.X, side=tk.TOP)
  1608. i=0
  1609. for k in PRESETS.val_presets:
  1610. if i%(8*8)==0 or i ==0:
  1611. c=0
  1612. b = tk.Label(frame,bg="black", text="X" )
  1613. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1614. r+=1
  1615. c=0
  1616. b = tk.Button(frame,bg="lightblue", text="EXEC " )
  1617. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1618. c+=1
  1619. b = tk.Button(frame,bg="lightblue", text="PAGE " + str(int(i/(8*8))+1) )
  1620. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1621. c+=1
  1622. b = tk.Button(frame,bg="lightblue", text="<NAME>" )
  1623. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1624. r+=1
  1625. c=0
  1626. i+=1
  1627. v=0
  1628. label = ""
  1629. #if k in PRESETS.label_presets:
  1630. # label = PRESETS.label_presets[k]
  1631. # #print([label])
  1632. sdata=PRESETS.val_presets[k]
  1633. BTN="go"
  1634. if "CFG" in sdata:#["BUTTON"] = "GO"
  1635. if "BUTTON" in sdata["CFG"]:
  1636. BTN = sdata["CFG"]["BUTTON"]
  1637. txt=str(k+1)+":"+str(BTN)+":"+str(len(sdata)-1)+"\n"+label
  1638. b = tk.Button(frame,bg="grey", text=txt,width=8,height=2)
  1639. b.bind("<Button>",Xevent(fix=0,elem=b,attr=k,data=self,mode="PRESET").cb)
  1640. b.bind("<ButtonRelease>",Xevent(fix=0,elem=b,attr=k,data=self,mode="PRESET").cb)
  1641. if k not in self.elem_presets:
  1642. self.elem_presets[k] = b
  1643. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1644. c+=1
  1645. if c >=8:
  1646. c=0
  1647. r+=1
  1648. self.refresh_exec()
  1649. def draw_input(self):
  1650. i=0
  1651. c=0
  1652. r=0
  1653. frame = tk.Frame(root2,bg="black")
  1654. frame.pack(fill=tk.X, side=tk.TOP)
  1655. b = tk.Label(frame,bg="black", text="------------------------ ---------------------------------------")
  1656. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1657. r=0
  1658. frame = tk.Frame(root2,bg="black")
  1659. frame.pack(fill=tk.X, side=tk.TOP)
  1660. b = tk.Label(frame, text="send:")
  1661. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1662. c+=1
  1663. b = tk.Entry(frame,bg="grey", text="",width=50)
  1664. self.entry = b
  1665. b.bind("<Button>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT").cb)
  1666. b.bind("<Key>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT").cb)
  1667. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1668. b.insert("end","d0:127,fx241:sinus:50:50:10,fx243:cosinus:50:50:10,d201:127,fx201:sinus:50:300:10")
  1669. r+=1
  1670. b = tk.Entry(frame,bg="grey", text="",width=20)
  1671. self.entry2 = b
  1672. b.bind("<Button>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT2").cb)
  1673. b.bind("<Key>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT2").cb)
  1674. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1675. b.insert("end","d1:0:4")
  1676. r+=1
  1677. b = tk.Entry(frame,bg="grey", text="",width=20)
  1678. self.entry3 = b
  1679. b.bind("<Button>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT3").cb)
  1680. #b.bind("<B1-Motion>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT3").cb)
  1681. b.bind("<Key>",Xevent(fix=0,elem=b,attr="INPUT",data=self,mode="INPUT3").cb)
  1682. b.grid(row=r, column=c, sticky=tk.W+tk.E)
  1683. b.insert("end","fx:alloff:::")
  1684. def draw_colorpicker(self,xframe):
  1685. import lib.colorpicker as colp
  1686. class _CB():
  1687. def __init__(self):
  1688. self.old_color = (0,0,0)
  1689. def cb(self,event,data):
  1690. cprint("colorpicker CB")
  1691. if "color" in data and self.old_color != data["color"] or event.num==2:
  1692. self.old_color = data["color"]
  1693. else:
  1694. return 0
  1695. color = data["color"]
  1696. print("e",event,data)
  1697. print("e",dir(event))#.keys())
  1698. try:
  1699. print("e.state",event.state)
  1700. except:pass
  1701. set_fade = FADE.val() #fade
  1702. if "color" in data and (event.num == 1 or event.num == 3 or event.num==2 or event.state in [256,1024]):
  1703. cr=None
  1704. cg=None
  1705. cb=None
  1706. if event.num == 1:
  1707. set_fade=FADE.val() #fade
  1708. cr = color[0]
  1709. cg = color[1]
  1710. cb = color[2]
  1711. elif event.num == 3:
  1712. cr = color[0]
  1713. cg = color[1]
  1714. cb = color[2]
  1715. set_fade=0
  1716. elif event.num == 2:
  1717. cr= "click"
  1718. cg= "click"
  1719. cb= "click"
  1720. elif event.state == 256:
  1721. cr = color[0]
  1722. cg = color[1]
  1723. cb = color[2]
  1724. set_fade=0
  1725. else:
  1726. set_fade=0
  1727. if cr is not None:
  1728. FIXTURES.encoder(fix=0,attr="RED",xval=cr,xfade=set_fade)
  1729. if cg is not None:
  1730. FIXTURES.encoder(fix=0,attr="GREEN",xval=cg,xfade=set_fade)
  1731. if cb is not None:
  1732. FIXTURES.encoder(fix=0,attr="BLUE",xval=cb,xfade=set_fade)
  1733. master.refresh_fix()
  1734. print("PICK COLOR:",data["color"])
  1735. _cb=_CB()
  1736. colp.colorpicker(xframe,width=600,height=100, xcb=_cb.cb)
  1737. return 0
  1738. canvas=tk.Canvas(xframe,width=600,height=100)
  1739. canvas["bg"] = "yellow" #"green"
  1740. canvas.pack()
  1741. # RGB
  1742. x=0
  1743. y=0
  1744. j=0
  1745. d = 20
  1746. for i in range(0,d+1):
  1747. fi = int(i*255/d)
  1748. f = 255-fi
  1749. if i > d/2:
  1750. pass#break
  1751. color = '#%02x%02x%02x' % (f, fi, fi)
  1752. print( "farbe", i*10, j, f,fi,fi,color)
  1753. r = canvas.create_rectangle(x, y, x+20, y+20, fill=color)
  1754. x+=20
  1755. def render(self):
  1756. #Xroot.bind("<Key>",Xevent(fix=0,elem=None,attr="ROOT",data=self,mode="ROOT").cb)
  1757. self.draw_input()
  1758. def ScrollFrame(root,width=50,height=100,bd=1):
  1759. #print("ScrollFrame init",width,height)
  1760. aframe=tk.Frame(root,relief=tk.GROOVE)#,width=width,height=height,bd=bd)
  1761. #aframe.place(x=0,y=0)
  1762. aframe.pack(side="left",fill="both",expand=1) #x=0,y=0)
  1763. canvas=tk.Canvas(aframe,width=width-24,height=height)
  1764. canvas["bg"] = "black" #"green"
  1765. bframe=tk.Frame(canvas)#,width=width,height=height)
  1766. bframe["bg"] = "blue"
  1767. scrollbar=tk.Scrollbar(aframe,orient="vertical",command=canvas.yview,width=20)
  1768. canvas.configure(yscrollcommand=scrollbar.set)
  1769. scrollbar.pack(side="right",fill="y")
  1770. canvas.pack(side="left",expand=1,fill="both")
  1771. canvas.create_window((0,0),window=bframe,anchor='nw')
  1772. bframe.bind("<Configure>",scroll(canvas).config)
  1773. canvas.bind("<Button>",Event("XXX").event)
  1774. canvas.bind("<Key>",Event("XXX").event)
  1775. canvas.bind("<KeyRelease>",Event("XXX").event)
  1776. return bframe
  1777. #frame = ScrollFrame(root)
  1778. class GUIHandler():
  1779. def __init__(self):
  1780. pass
  1781. def update(self,fix,attr,args={}):
  1782. print("GUIHandler.update()",fix,attr,args)
  1783. for i,k in enumerate(args):
  1784. v = args[k]
  1785. #print("GUI-H", i,k,v)
  1786. class Fixtures(Base):
  1787. def __init__(self):
  1788. super().__init__()
  1789. #self.load()
  1790. self.fixtures = OrderedDict()
  1791. self.gui = GUIHandler()
  1792. def load_patch(self):
  1793. filename="patch"
  1794. d,l = self._load(filename)
  1795. self.fixtures = OrderedDict()
  1796. for i in l:
  1797. sdata = d[i]
  1798. for attr in sdata["ATTRIBUT"]:
  1799. sdata["ATTRIBUT"][attr]["ACTIVE"] = 0
  1800. #print("load",filename,sdata)
  1801. #if "CFG" not in sdata:
  1802. # sdata["CFG"] = OrderedDict()
  1803. self.fixtures[str(i)] = sdata
  1804. #PRESETS.label_presets = l
  1805. def backup_patch(self):
  1806. filename = "patch"
  1807. data = self.fixtures
  1808. labels = {}
  1809. for k in data:
  1810. labels[k] = k
  1811. self._backup(filename,data,labels)
  1812. def fx_off(self,fix=None):
  1813. if not fix or fix == "all":
  1814. #self.data.elem_fx_commands[self.attr]["bg"] = "magenta"
  1815. for fix in self.fixtures:
  1816. data = self.fixtures[fix]
  1817. for attr in data["ATTRIBUT"]:
  1818. data["ATTRIBUT"][attr]["FX"] = ""
  1819. def get_attr(self,fix,attr):
  1820. if fix in self.fixtures:
  1821. data = self.fixtures[fix]
  1822. if "ATTRIBUT" in data:
  1823. if attr in data["ATTRIBUT"]:
  1824. return data["ATTRIBUT"][attr]
  1825. def get_dmx(self,fix,attr):
  1826. #cprint("get_dmx",[fix,attr])
  1827. if fix in self.fixtures:
  1828. data = self.fixtures[fix]
  1829. if "DMX" in data:
  1830. DMX = int(data["DMX"])
  1831. else:
  1832. return -1
  1833. if "UNIVERS" in data:
  1834. DMX += (int(data["UNIVERS"])*512)
  1835. adata = self.get_attr(fix,attr)
  1836. #-hier ende 8.2.22
  1837. #cprint("adata",adata,DMX)
  1838. if adata:
  1839. if "NR" in adata:
  1840. NR = adata["NR"]
  1841. if NR:
  1842. DMX+=NR-1
  1843. else:
  1844. return -2
  1845. return DMX
  1846. return -4
  1847. return -3
  1848. def update_raw(self,rdata,update=1):
  1849. cprint("update_raw",len(rdata))
  1850. cmd = []
  1851. for i,d in enumerate(rdata):
  1852. xcmd = {"DMX":""}
  1853. #print("fix:",i,d)
  1854. fix = d["FIX"]
  1855. attr = d["ATTR"]
  1856. v2 = d["VALUE"]
  1857. v2_fx = d["FX"]
  1858. if fix not in self.fixtures:
  1859. continue
  1860. sdata = self.fixtures[fix] #shortcat
  1861. ATTR = sdata["ATTRIBUT"]
  1862. sDMX = 0
  1863. if sdata["DMX"] > 0:
  1864. #print( sdata)
  1865. sDMX = (sdata["UNIVERS"]*512)+sdata["DMX"]
  1866. #sDMX =sdata["DMX"]
  1867. #else:
  1868. # continue
  1869. if attr not in ATTR:
  1870. continue
  1871. #DMX = FIXTURES.get_dmx(fix)
  1872. if ATTR[attr]["NR"] >= 0:
  1873. DMX = sDMX+ATTR[attr]["NR"]-1
  1874. xcmd["DMX"] = str(DMX)
  1875. else:
  1876. if attr == "DIM" and ATTR[attr]["NR"] < 0:
  1877. xcmd["VIRTUAL"] = {}
  1878. for a in ATTR:
  1879. if ATTR[a]["MASTER"]:
  1880. xcmd["VIRTUAL"][a] = sDMX+ATTR[a]["NR"]-1
  1881. #print( "VIRTUAL",xcmd)
  1882. cmd.append(xcmd)
  1883. v=ATTR[attr]["VALUE"]
  1884. if v2 is not None and update:
  1885. ATTR[attr]["VALUE"] = v2
  1886. #self.data.elem_attr[fix][attr]["text"] = str(attr)+' '+str(round(v,2))
  1887. text = str(attr)+' '+str(round(v,2))
  1888. #self.gui.update(fix,attr,args={"text":text})
  1889. #print("END 5454 _=_=_=_=_==_")
  1890. cprint("update_raw",cmd,color="red")
  1891. return cmd
  1892. def encoder(self,fix,attr,xval="",xfade=0):
  1893. cprint("FIXTURES.encoder",fix,attr,xval,xfade,color="yellow")
  1894. if attr == "CLEAR":
  1895. self.clear()
  1896. return 0
  1897. if fix not in self.fixtures:
  1898. jdata=[{"MODE":"---"}]
  1899. ii =0
  1900. jclient_send(jdata)
  1901. for fix in self.fixtures:
  1902. ii+=1
  1903. #cprint(fix,attr,xval)
  1904. data = self.fixtures[fix]
  1905. if attr in data["ATTRIBUT"]:
  1906. if xval == "click":
  1907. self.select(fix,attr,mode="on")
  1908. elif data["ATTRIBUT"][attr]["ACTIVE"]:
  1909. if fix: # prevent endless recursion
  1910. self.encoder(fix,attr,xval,xfade)
  1911. jdata=[{"MODE":ii}]
  1912. jclient_send(jdata)
  1913. return 0
  1914. data = self.fixtures[fix]
  1915. if xval == "click":
  1916. #cprint(data)
  1917. return self.select(fix,attr,mode="toggle")
  1918. v2=data["ATTRIBUT"][attr]["VALUE"]
  1919. change=0
  1920. increment = 4.11
  1921. jdata = {"MODE":"ENC"}
  1922. if xval == "+":
  1923. v2+= increment
  1924. jdata["INC"] = increment
  1925. change=1
  1926. elif xval == "-":
  1927. jdata["INC"] = increment*-1
  1928. v2-= increment
  1929. change=1
  1930. elif type(xval) is int or type(xval) is float:
  1931. v2 = xval
  1932. change=1
  1933. if v2 < 0:
  1934. v2=0
  1935. elif v2 > 256:
  1936. v2=256
  1937. jdata["VALUE"] = round(v2,4)
  1938. jdata["FIX"] = fix
  1939. jdata["ATTR"] = attr
  1940. jdata["DMX"] = FIXTURES.get_dmx(fix,attr)
  1941. out = {}
  1942. if change:
  1943. data["ATTRIBUT"][attr]["ACTIVE"] = 1
  1944. data["ATTRIBUT"][attr]["VALUE"] = round(v2,4)
  1945. jdata["FADE"] = 0
  1946. if xfade:
  1947. jdata["FADE"] = xfade
  1948. if not modes.val("BLIND"):
  1949. jdata = [jdata]
  1950. print(jdata)
  1951. jclient_send(jdata)
  1952. return v2
  1953. def get_active(self):
  1954. cprint("get_active",self,"get_active")
  1955. CFG = OrderedDict()
  1956. sdata = OrderedDict()
  1957. sdata["CFG"] = CFG # OrderedDict()
  1958. sdata["CFG"]["FADE"] = FADE.val()
  1959. sdata["CFG"]["DEALY"] = 0
  1960. #sdata["CFG"]["BUTTON"] = "GO"
  1961. for fix in self.fixtures:
  1962. data = self.fixtures[fix]
  1963. for attr in data["ATTRIBUT"]:
  1964. if data["ATTRIBUT"][attr]["ACTIVE"]:
  1965. if fix not in sdata:
  1966. sdata[fix] = {}
  1967. if attr not in sdata[fix]:
  1968. sdata[fix][attr] = OrderedDict()
  1969. if not modes.val("STONY_FX"):
  1970. sdata[fix][attr]["VALUE"] = data["ATTRIBUT"][attr]["VALUE"]
  1971. #sdata[fix][attr]["FADE"] = FADE.val() #fade
  1972. else:
  1973. sdata[fix][attr]["VALUE"] = None #data["ATTRIBUT"][attr]["VALUE"]
  1974. if "FX" not in data["ATTRIBUT"][attr]:
  1975. data["ATTRIBUT"][attr]["FX"] =""
  1976. if "FX2" not in data["ATTRIBUT"][attr]:
  1977. data["ATTRIBUT"][attr]["FX2"] ={}
  1978. sdata[fix][attr]["FX"] = data["ATTRIBUT"][attr]["FX"]
  1979. sdata[fix][attr]["FX2"] = data["ATTRIBUT"][attr]["FX2"]
  1980. return sdata
  1981. def select(self,fix=None,attr=None,mode="on"):
  1982. cprint("FIXTURES.select()",fix,attr,mode,color="yellow")
  1983. out = 0
  1984. if fix in self.fixtures:
  1985. data = self.fixtures[fix]
  1986. if attr in data["ATTRIBUT"]:
  1987. if mode == "on":
  1988. if not data["ATTRIBUT"][attr]["ACTIVE"]:
  1989. data["ATTRIBUT"][attr]["ACTIVE"] = 1
  1990. out = 1
  1991. elif mode == "off":
  1992. if data["ATTRIBUT"][attr]["ACTIVE"]:
  1993. data["ATTRIBUT"][attr]["ACTIVE"] = 0
  1994. out = 1
  1995. elif mode == "toggle":
  1996. if data["ATTRIBUT"][attr]["ACTIVE"]:
  1997. data["ATTRIBUT"][attr]["ACTIVE"] = 0
  1998. else:
  1999. data["ATTRIBUT"][attr]["ACTIVE"] = 1
  2000. out = 1
  2001. return out
  2002. def clear(self):
  2003. out = 0
  2004. for fix in self.fixtures:
  2005. data = self.fixtures[fix]
  2006. for attr in data["ATTRIBUT"]:
  2007. if attr.endswith("-FINE"):
  2008. continue
  2009. if data["ATTRIBUT"][attr]["ACTIVE"]:
  2010. out +=1
  2011. data["ATTRIBUT"][attr]["ACTIVE"] = 0
  2012. return out
  2013. class Presets(Base):
  2014. def __init__(self):
  2015. super().__init__()
  2016. #self.load()
  2017. self._last_copy = None
  2018. self._last_move = None
  2019. def load_presets(self):
  2020. filename="presets"
  2021. d,l = self._load(filename)
  2022. for i in d:
  2023. sdata = d[i]
  2024. if "CFG" not in sdata:
  2025. sdata["CFG"] = OrderedDict()
  2026. if "FADE" not in sdata["CFG"]:
  2027. sdata["CFG"]["FADE"] = 4
  2028. if "DELAY" not in sdata["CFG"]:
  2029. sdata["CFG"]["DELAY"] = 0
  2030. if "BUTTON" not in sdata["CFG"]:
  2031. sdata["CFG"]["BUTTON"] = "GO"
  2032. self.val_presets = d
  2033. self.label_presets = l
  2034. def check_cfg(self,nr=None):
  2035. cprint("PRESETS.check_cfg()",nr)
  2036. ok = 0
  2037. if nr is not None:
  2038. ok += self._check_cfg(nr)
  2039. else:
  2040. for nr in self.val_presets:
  2041. ok += self._check_cfg(nr)
  2042. return ok
  2043. def _check_cfg(self,nr):
  2044. #cprint("PRESETS._check_cfg()",nr)
  2045. ok=0
  2046. if nr in self.val_presets:
  2047. sdata = self.val_presets[nr]
  2048. if "CFG" not in sdata:
  2049. sdata["CFG"] = OrderedDict()
  2050. ok += 1
  2051. if "FADE" not in sdata["CFG"]:
  2052. sdata["CFG"]["FADE"] = 4
  2053. ok += 1
  2054. if "DELAY" not in sdata["CFG"]:
  2055. sdata["CFG"]["DELAY"] = 0
  2056. ok += 1
  2057. if "BUTTON" not in sdata["CFG"]:
  2058. sdata["CFG"]["BUTTON"] = "GO"
  2059. ok += 1
  2060. if ok:
  2061. cprint("REPAIR CFG's",nr,sdata["CFG"],color="red")
  2062. else:
  2063. cprint("nr not in data ",nr,color="red")
  2064. return ok
  2065. def backup_presets(self):
  2066. filename = "presets"
  2067. data = self.val_presets
  2068. labels = self.label_presets
  2069. self._backup(filename,data,labels)
  2070. def get_cfg(self,nr):
  2071. cprint("PRESETS.get_cfg()",nr)
  2072. self.check_cfg(nr)
  2073. if nr not in self.val_presets:
  2074. cprint("get_cfg",self,"error get_cfg no nr:",nr,color="red")
  2075. return {}
  2076. if "CFG" in self.val_presets[nr]:
  2077. return self.val_presets[nr]["CFG"]
  2078. def clean(self,nr):
  2079. if nr not in self.val_presets:
  2080. self.val_presets[nr] = OrderedDict()
  2081. #self.val_presets[nr]["VALUE"] = 0
  2082. #self.val_presets[nr]["FX"] = ""
  2083. sdata = self.val_presets[nr]
  2084. for fix in sdata:
  2085. print("exec.clear()",nr,fix,sdata[fix])
  2086. for attr in sdata[fix]:
  2087. row = sdata[fix][attr]
  2088. if fix == "CFG":
  2089. continue
  2090. if "VALUE" not in row:
  2091. row["VALUE"] = None
  2092. if "FX" not in row:
  2093. row["FX"] = ""
  2094. if "FX2" not in row:
  2095. row["FX2"] = OrderedDict()
  2096. elif row["FX2"]:
  2097. for k in ["SIZE","SPEED","START","OFFSET"]:
  2098. row["FX2"][k] = int( row["FX2"][k] )
  2099. row["FX"] = ""
  2100. if "FX" in row and row["FX"] and not row["FX2"]: # rebuild old FX to Dict-FX2
  2101. #"off:0:0:0:16909:-:"
  2102. x = row["FX"].split(":")
  2103. print(x,len(x))
  2104. #'FX2': {'TYPE': 'sinus', 'SIZE': 200, 'SPEED': 30, 'START': 0, 'OFFSET': 2805, 'BASE': '-'}}
  2105. if len(x) >= 6:
  2106. row["FX2"]["TYPE"] = x[0]
  2107. row["FX2"]["SIZE"] = int(x[1])
  2108. row["FX2"]["SPEED"] = int(x[2])
  2109. row["FX2"]["START"] = int(x[3])
  2110. row["FX2"]["OFFSET"] = int(x[4])
  2111. row["FX2"]["BASE"] = x[5]
  2112. row["FXOLD"] = row["FX"]
  2113. row["FX"] = ""
  2114. cprint("exec.clear()",nr,fix,row)
  2115. def get_raw_map(self,nr):
  2116. self.clean(nr)
  2117. print("get_raw_map",nr)
  2118. sdata = self.val_presets[nr]
  2119. cmd = ""
  2120. out = []
  2121. dmx=-1
  2122. for fix in sdata:
  2123. if fix == "CFG":
  2124. #print("CFG",nr,sdata[fix])
  2125. continue
  2126. for attr in sdata[fix]:
  2127. x = {}
  2128. #print("RAW",attr)
  2129. x["FIX"] = fix
  2130. x["ATTR"] = attr
  2131. x["VALUE"] = sdata[fix][attr]["VALUE"]
  2132. x["FX"] = sdata[fix][attr]["FX"]
  2133. x["FX2"] = sdata[fix][attr]["FX2"]
  2134. #x["DMX"] = sdata[fix][attr]["NR"]
  2135. out.append(x)
  2136. return out
  2137. def get_btn_txt(self,nr):
  2138. sdata=self.val_presets[nr]
  2139. BTN="go"
  2140. if "CFG" in sdata:
  2141. if "BUTTON" in sdata["CFG"]:
  2142. BTN = sdata["CFG"]["BUTTON"]
  2143. _label = self.label_presets[nr] # = label
  2144. txt=str(nr+1)+":"+str(BTN)+":"+str(len(sdata)-1)+"\n"+_label
  2145. print("get_btn_txt",nr,[txt])
  2146. return txt
  2147. def btn_cfg(self,nr,txt=None):
  2148. if nr not in self.val_presets:
  2149. return ""
  2150. if "CFG" not in self.val_presets[nr]:
  2151. self.val_presets[nr]["CFG"] = OrderedDict()
  2152. if "BUTTON" not in self.val_presets[nr]["CFG"]:
  2153. self.val_presets[nr]["CFG"]["BUTTON"] = ""
  2154. if type(txt) is str:
  2155. self.val_presets[nr]["CFG"]["BUTTON"] = txt
  2156. if self.val_presets[nr]["CFG"]["BUTTON"] is None:
  2157. self.val_presets[nr]["CFG"]["BUTTON"] = ""
  2158. print("EEE", self.val_presets[nr]["CFG"]["BUTTON"] )
  2159. return self.val_presets[nr]["CFG"]["BUTTON"]
  2160. def label(self,nr,txt=None):
  2161. if nr not in self.label_presets:
  2162. return ""
  2163. if type(txt) is str:
  2164. self.label_presets[nr] = txt
  2165. print("set label",nr,[txt])
  2166. print("??? ?? set label",nr,[txt])
  2167. return self.label_presets[nr]
  2168. def clear_move(self):
  2169. cprint("PRESETS.clear_move()",end=" ")
  2170. self.clear_copy()
  2171. def clear_copy(self):
  2172. cprint("PRESETS.clear_copy()",end=" ")
  2173. if self._last_copy is not None:
  2174. cprint("=OK=",color="red")
  2175. self._last_copy = None
  2176. else:
  2177. cprint("=NONE=",color="green")
  2178. def copy(self,nr,overwrite=1):
  2179. cprint("PRESETS._copy",nr,"last",self._last_copy)
  2180. if nr:
  2181. if self._last_copy is not None:
  2182. ok = self._copy(self._last_copy,nr,overwrite=overwrite)
  2183. return ok #ok
  2184. else:
  2185. self._last_copy = nr
  2186. cprint("PRESETS.copy START ",color="red")
  2187. return 0
  2188. return 1 # on error reset move
  2189. def _copy(self,nr_from,nr_to,overwrite=1):
  2190. cprint("PRESETS._copy",nr_from,"to",nr_to)
  2191. self.check_cfg(nr_from)
  2192. if self._last_copy is None:
  2193. cprint("PRESETS._copy last nr is None")
  2194. return 0
  2195. if nr_from in self.val_presets and nr_to in self.val_presets:
  2196. fdata = self.val_presets[nr_from]
  2197. tdata = self.val_presets[nr_to]
  2198. #cprint(fdata)
  2199. flabel = self.label_presets[nr_from]
  2200. tlabel = self.label_presets[nr_to]
  2201. self.val_presets[nr_to] = copy.deepcopy(fdata)
  2202. self.label_presets[nr_to] = flabel
  2203. if not overwrite: #default
  2204. cprint("overwrite",overwrite)
  2205. self.val_presets[nr_from] = copy.deepcopy(tdata)
  2206. self.label_presets[nr_from] = tlabel
  2207. #self.label_presets[nr_from] = "MOVE"
  2208. self.clear_copy()
  2209. cprint("PRESETS.copy OK",color="red")
  2210. return 1
  2211. def move(self,nr):
  2212. cprint("PRESETS.move",self._last_copy,"to",nr)
  2213. if nr:
  2214. last = self._last_copy
  2215. ok= self.copy(nr,overwrite=0)
  2216. if ok and last:
  2217. cprint("PRESETS.move OK",color="red")
  2218. #self.delete(last)
  2219. return ok #ok
  2220. return 0 # on error reset move
  2221. def delete(self,nr):
  2222. cprint("PRESETS.delete",nr)
  2223. ok=0
  2224. if nr in self.val_presets:
  2225. self.val_presets[nr] = OrderedDict()
  2226. self.label_presets[nr] = ""
  2227. ok = 1
  2228. self.check_cfg(nr)
  2229. return ok
  2230. def rec(self,nr,data,arg=""):
  2231. print("rec",self,"rec()",data,arg)
  2232. self.check_cfg(nr)
  2233. self.val_presets[nr] = data
  2234. return 1
  2235. class GUI_grid():
  2236. def __init__(self,root,data,title="tilte",width=800):
  2237. self.data = data
  2238. self.frame = tk.Frame(root,bg="black",width=width)
  2239. self.frame.pack(fill=tk.BOTH, side=tk.LEFT)
  2240. r=0
  2241. c=0
  2242. i=1
  2243. for row in data:
  2244. self.b = tk.Button(self.frame,bg="lightblue", text=row["text"],width=11,height=4)
  2245. #self.b.bind("<Button>",Xevent(fix=fix,elem=b).cb)
  2246. self.b.grid(row=r, column=c, sticky=tk.W+tk.E)#,anchor="w")
  2247. c+=1
  2248. if c % 8 == 0:
  2249. r+=1
  2250. c=0
  2251. i+=1
  2252. self.frame.pack()
  2253. class BEvent():
  2254. def __init__(self,data,cb):
  2255. self._data = data
  2256. self._cb = cb
  2257. def cb(self,event):
  2258. #print(self,event)
  2259. self._cb(event,self._data)
  2260. class GUI_menu():
  2261. def __init__(self,root,data,title="tilte",width=800):
  2262. global tk
  2263. self.data = data
  2264. self.frame = tk.Frame(root,bg="black",width=width)
  2265. self.frame.pack(fill=tk.BOTH, side=tk.LEFT)
  2266. r=0
  2267. c=0
  2268. i=1
  2269. self.b = tk.Label(self.frame,bg="lightblue", text="MAIN:MENU",width=10,height=1)
  2270. self.b.grid(row=r, column=c, sticky=tk.W+tk.E)#,anchor="w")
  2271. r+=1
  2272. for row in data:
  2273. #print(i)
  2274. #row = data[i]
  2275. self.b = tk.Button(self.frame,bg="lightblue", text=row["text"],width=10,height=3)
  2276. self.b.bind("<Button>",BEvent({"NR":i,"text":row["text"]},self.callback).cb)
  2277. self.b.grid(row=r, column=c, sticky=tk.W+tk.E)#,anchor="w")
  2278. r+=1
  2279. i+=1
  2280. self.frame.pack()
  2281. def callback(self,event,data={}):
  2282. print("callback543",self,event,data)
  2283. window_manager.top(data["text"])# = WindowManager()
  2284. lf_nr = 0
  2285. class GUIWindow():
  2286. def __init__(self,title="tilte",master=0,width=100,height=100,left=None,top=None):
  2287. global lf_nr
  2288. if master:
  2289. self.tk = tkinter.Tk()
  2290. defaultFont = tkinter.font.nametofont("TkDefaultFont")
  2291. print(defaultFont)
  2292. defaultFont.configure(family="FreeSans",
  2293. size=10,
  2294. weight="bold")
  2295. #self.tk.option_add("*Font", FontBold)
  2296. else:
  2297. self.tk = tkinter.Toplevel()
  2298. #print(title,self.tk.__doc__)
  2299. self.tk["bg"] = "black"
  2300. self.tk.bind("<Button>",self.callback)
  2301. self.tk.bind("<Key>",self.callback)
  2302. self.tk.bind("<KeyRelease>",self.callback)
  2303. self.tk.title(""+str(title)+" "+str(lf_nr)+":"+str(rnd_id))
  2304. lf_nr+=1
  2305. #self.tk.geometry("270x600+0+65")
  2306. geo ="{}x{}".format(width,height)
  2307. if left is not None:
  2308. geo += "+{}".format(left)
  2309. if top is not None:
  2310. geo += "+{}".format(top)
  2311. #self._event_clear = Xevent(fix=0,elem=None,attr="CLEAR",data=self,mode="ROOT").cb
  2312. self.tk.geometry(geo)
  2313. def title(self,title=None):
  2314. if title is None:
  2315. return self.tk.title()
  2316. else:
  2317. return self.tk.title(title)
  2318. def show(self):
  2319. pass
  2320. def mainloop(self):
  2321. try:
  2322. self.tk.mainloop()
  2323. finally:
  2324. self.tk.quit()
  2325. def callback(self,event,data={}):#value=255):
  2326. print()
  2327. print()
  2328. cprint("<GUI>",event,color="yellow")
  2329. cprint("<GUI>",event.state,data,[event.type],color="yellow")
  2330. value = 255
  2331. if "Release" in str(event.type) or str(event.type) == '5' or str(event.type) == '3':
  2332. value = 0
  2333. if "keysym" in dir(event):
  2334. if "Escape" == event.keysym:
  2335. FIXTURES.clear()
  2336. modes.val("ESC",1)
  2337. master.refresh_fix()
  2338. elif event.keysym in "ebfclrms" and value:
  2339. if "e" == event.keysym:
  2340. modes.val("EDIT",1)
  2341. elif "b" == event.keysym:
  2342. modes.val("BLIND",1)
  2343. elif "f" == event.keysym:
  2344. modes.val("FLASH",1)
  2345. elif "c" == event.keysym:
  2346. modes.val("CFG-BTN",1)
  2347. elif "l" == event.keysym:
  2348. modes.val("LABEL",1)
  2349. elif "r" == event.keysym:
  2350. modes.val("REC",1)
  2351. elif "m" == event.keysym:
  2352. x=modes.val("MOVE",1)
  2353. if not x:
  2354. PRESETS.clear_move()
  2355. elif "s" == event.keysym:
  2356. modes.val("SELECT",1)
  2357. elif event.keysym in ["1","2","3","4","5","6","7","8","9","0"]:
  2358. nr = int( event.keysym)
  2359. if nr == 0:
  2360. nr =10
  2361. cprint("F-KEY",value,nr)
  2362. xfade = 0
  2363. if FADE._is():
  2364. xfade = 0
  2365. master.preset_go(128-1+nr,xfade=xfade,val=value)
  2366. elif event.keysym in ["F1","F2","F3","F4","F5","F6","F7","F8","F9","F10","F11","F12"]:
  2367. nr = int( event.keysym[1])-1
  2368. cprint("F-KEY",value,nr)
  2369. xfade = 0
  2370. if FADE._is():
  2371. xfade = 0
  2372. master.preset_go(65-1+nr,xfade=xfade,val=value)
  2373. elif "End" == event.keysym:
  2374. FIXTURES.fx_off("all")
  2375. CONSOLE.fx_off("all")
  2376. CONSOLE.flash_off("all")
  2377. elif "Delete" == event.keysym:
  2378. #PRESETS.delete(nr)
  2379. if value:
  2380. modes.val("DEL",1)
  2381. class WindowManager():
  2382. def __init__(self):
  2383. self.windows = {}
  2384. self.nr= 0
  2385. self.first=""
  2386. def new(self,w,name=""):
  2387. if not self.first:
  2388. if name:
  2389. self.first = name
  2390. else:
  2391. self.first = str(self.nr)
  2392. w.tk.attributes('-topmost',True)
  2393. if name:
  2394. self.windows[str(name)] = w
  2395. else:
  2396. self.windows[str(self.nr)] = w
  2397. self.nr+=1
  2398. #w.show()
  2399. def mainloop(self):
  2400. self.windows[self.first].mainloop()
  2401. def top(self,name):
  2402. name = str(name)
  2403. if name in self.windows:
  2404. self.windows[name].tk.attributes('-topmost',True)
  2405. self.windows[name].tk.attributes('-topmost',False)
  2406. else:
  2407. print(name,"not in self.windows",self.windows.keys())
  2408. class Console():
  2409. def __init__(self):
  2410. pass
  2411. def flash_off(self,fix):
  2412. pass#client.send("df0:alloff:::,")
  2413. def fx_off(self,fix):
  2414. cprint("Console.fx_off()",fix)
  2415. if not fix or fix == "all":
  2416. #client.send("fx0:alloff:,fxf:alloff:,")
  2417. #client.send("df0:alloff:::,")
  2418. j = []
  2419. if 0:
  2420. jdata = {'VALUE': None, 'args': [], 'FX': 'alloff::::', 'FADE': 2, 'DMX': '0'}
  2421. j.append(jdata)
  2422. jdata = {'VALUE': None, 'args': [], 'FX': 'alloff::::', 'FADE': 2,'FLASH':1, 'DMX': '0'}
  2423. j.append(jdata)
  2424. else:
  2425. jdata = {'VALUE': None, 'args': [], 'FX2': {"TYPE":"alloff"}, 'FADE': 2,'FLASH':1, 'DMX': '1'}
  2426. j.append(jdata)
  2427. jclient_send(j)
  2428. return 0
  2429. window_manager = WindowManager()
  2430. CONSOLE = Console()
  2431. PRESETS = Presets()
  2432. PRESETS.load_presets()
  2433. FIXTURES = Fixtures()
  2434. FIXTURES.load_patch()
  2435. master = GUI()
  2436. w = GUIWindow("MAIN",master=1,width=100,height=450,left=0,top=65)
  2437. data = []
  2438. #data.append({"text":"COMMAND"})
  2439. data.append({"text":"EXEC"})
  2440. data.append({"text":"DIMMER"})
  2441. data.append({"text":"FIXTURES"})
  2442. f = GUI_menu(w.tk,data)
  2443. window_manager.new(w)
  2444. name="EXEC"
  2445. w = GUIWindow(name,master=0,width=800,height=400,left=110,top=65)
  2446. w1 = ScrollFrame(w.tk,width=800,height=400)
  2447. #frame_exe = w.tk
  2448. master.draw_preset(w1)#w.tk)
  2449. window_manager.new(w,name)
  2450. name="DIMMER"
  2451. w = GUIWindow(name,master=0,width=800,height=400,left=110,top=65)
  2452. w2 = ScrollFrame(w.tk,width=800,height=400)
  2453. #frame_dim = w1 # w.tk
  2454. #master.draw_dim(w1.tk)
  2455. window_manager.new(w,name)
  2456. name="FIXTURES"
  2457. w = GUIWindow(name,master=0,width=800,height=400,left=110,top=65)
  2458. w1 = ScrollFrame(w.tk,width=800,height=400)
  2459. #frame_fix = w1 #w.tk
  2460. master.draw_fix(w1,w2)#.tk)
  2461. window_manager.new(w,name)
  2462. name="ENCODER"
  2463. ww = GUIWindow(name,master=0,width=800,height=50,left=110,top=500)
  2464. Xroot = ww.tk
  2465. w = None
  2466. root = tk.Frame(Xroot,bg="black",width="10px")
  2467. print("print pack",root)
  2468. root.pack(fill=tk.BOTH,expand=0, side=tk.LEFT)
  2469. root3 = tk.Frame(Xroot,bg="black",width="20px")
  2470. root3.pack(fill=tk.BOTH,expand=0, side=tk.LEFT)
  2471. root2 = tk.Frame(Xroot,bg="black",width="1px")
  2472. master.draw_enc(root2)
  2473. root2.pack(fill=tk.BOTH,expand=0, side=tk.LEFT)
  2474. name = "COMMAND"
  2475. w = GUIWindow(name,master=0,width=350,height=200,left=920,top=65)
  2476. master.draw_command(w.tk)
  2477. window_manager.new(w,name)
  2478. name="PATCH"
  2479. w = GUIWindow(name,master=0,width=800,height=400,left=110,top=65)
  2480. w1 = ScrollFrame(w.tk,width=800,height=400)
  2481. master.draw_patch(w1)
  2482. window_manager.new(w,name)
  2483. name="FX"
  2484. w = GUIWindow(name,master=0,width=350,height=250,left=920,top=305)
  2485. #frame_fx = w.tk
  2486. master.draw_fx(w.tk)
  2487. window_manager.new(w,name)
  2488. #LibreLightDesk
  2489. name="COLORPICKER"
  2490. w = GUIWindow(name,master=0,width=580,height=100,left=80,top=620)
  2491. master.draw_colorpicker(w.tk)
  2492. window_manager.new(w,name)
  2493. #Xroot = tk.Tk()
  2494. #Xroot["bg"] = "black" #white
  2495. #Xroot.title( xtitle+" "+str(rnd_id) )
  2496. #Xroot.geometry("1024x800+130+65")
  2497. master.render()
  2498. #w = frame_fix #GUIWindow("OLD",master=0,width=800,height=500,left=130,top=65)
  2499. window_manager.new(w,name)
  2500. try:
  2501. #root.mainloop()
  2502. #tk.mainloop()
  2503. window_manager.mainloop()
  2504. finally:
  2505. master.exit()