fix.py 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825
  1. #/usr/bin/python3
  2. import copy
  3. from collections import OrderedDict
  4. import tkinter
  5. tk = tkinter
  6. from idlelib.tooltip import Hovertip
  7. import tkgui.dialog as dialoglib
  8. dialog = dialoglib.Dialog()
  9. import __main__ as MAIN
  10. from lib.cprint import *
  11. import lib.fixlib as fixlib
  12. import lib.showlib as showlib
  13. import lib.libtk as libtk
  14. class BaseCallback():
  15. def __init__(self,cb=None,args={}):
  16. self._cb=cb
  17. self.args = args
  18. def cb(self,**args):
  19. print("BaseCallback.cb()")
  20. print(" ",self.args)
  21. print(" ",self._cb)
  22. if self._cb:
  23. if self.args:
  24. self._cb(args=self.args)
  25. else:
  26. self._cb()
  27. def GUI_LOAD_FIXTURE_LIST(frame,data={"EMPTY":"None"},cb=None,bg="black"):
  28. #print("__func__",__func__)
  29. print("#",sys._getframe().f_code.co_name)
  30. blist = data
  31. blist = blist[:100]
  32. frame.configure(bg=bg)
  33. # table header
  34. c=0
  35. for r,row in enumerate(blist):
  36. bg="lightgrey"
  37. dbg="grey"
  38. dbg="lightgrey"
  39. c+=1
  40. b = tk.Label(frame,bg="grey",text=str("Nr."))
  41. b.grid(row=r, column=c, sticky=tk.W) #+tk.E)
  42. c+=1
  43. for k,v in row.items():
  44. b = tk.Label(frame,bg="grey",text=k)
  45. b.grid(row=r, column=c, sticky=tk.W) #+tk.E)
  46. c+=1
  47. break
  48. if not cb:
  49. cb = DummyCallback()
  50. # table data
  51. for r,row in enumerate(blist):
  52. c=1
  53. bg="lightgrey"
  54. #dbg="grey"
  55. b = tk.Button(frame,text=r+1,anchor="w",bg=dbg,width=6,height=1,relief="sunken")
  56. b.grid(row=r+1, column=c, sticky=tk.W ) #+tk.E)
  57. c+=1
  58. bg="lightgrey"
  59. dbg="lightgrey"
  60. for k,v in row.items():
  61. #if v > time.strftime("%Y-%m-%d %X", time.localtime(time.time()-3600*4)):
  62. # dbg = "lightgreen"
  63. #elif v > time.strftime("%Y-%m-%d %X", time.localtime(time.time()-3600*24*7)):
  64. # dbg = "green"
  65. if "." in v:
  66. #v = v.split(".",-1)
  67. v = v[::1].split(".",1)[0]#[::-1]
  68. if c == 3:
  69. bg="grey"
  70. dbg="grey"
  71. _cb2 = BaseCallback(cb=cb,args={"key":k,"val":v,"data":row}).cb
  72. b = tk.Button(frame,text=v,anchor="w",height=1,bg=bg,command=_cb2)
  73. else:
  74. b = tk.Button(frame,text=v,anchor="w",bg=dbg,relief="flat",height=1)
  75. b.config(activebackground=dbg)
  76. b.grid(row=r+1, column=c, sticky=tk.W+tk.E)
  77. c+=1
  78. bg="lightgrey"
  79. dbg="lightgrey"
  80. class GUI_FixtureEditor():
  81. def __init__(self,root,frame,data,title="tilte",width=800):
  82. #xfont = tk.font.Font(family="FreeSans", size=5, weight="bold")
  83. self.font8 = ("FreeSans",8)
  84. self.dmx=1
  85. self.univ=0
  86. self.elem=[]
  87. self.fader_elem = []
  88. self.pw = None
  89. self.header=[]
  90. self.data = data
  91. self.fixture = []
  92. self.title = title
  93. self.width = width
  94. #cprint("GUI:",root,title)
  95. self.root = root
  96. self.frame = frame
  97. self.draw()
  98. def draw(self):
  99. root = self.frame
  100. title = self.title
  101. width = self.width
  102. data = self.data
  103. self.fader_elem = []
  104. #Head 1
  105. self.frame = tk.Frame(root,bg="grey",width=width)
  106. self.frame.pack(fill="both", side=tk.TOP)
  107. self.frame = tk.Frame(self.frame,bg="grey",width=width)
  108. self.frame.pack(fill="both", side=tk.RIGHT)#EFT)
  109. bg = "lightblue"
  110. self.b = tk.Button(self.frame,bg=bg,text="IMPORT FROM SHOW", width=20)#,command=self.event) #bv.change_dmx)
  111. self.b["command"] = self.open_fixture_list_import
  112. self.b.pack( side=tk.LEFT)
  113. self.b = tk.Button(self.frame,bg=bg,text="USER", width=5)#,command=self.event) #bv.change_dmx)
  114. self.b["command"] = self.open_fixture_list_user
  115. self.b.pack( side=tk.LEFT)
  116. self.b = tk.Button(self.frame,bg=bg,text="GLOBAL", width=5)#,command=self.event) #bv.change_dmx)
  117. self.b["command"] = self.open_fixture_list_global
  118. self.b.pack( side=tk.LEFT)
  119. self.b = tk.Label(self.frame,bg=bg ,text="")
  120. self.b.pack(fill=None, side=tk.LEFT)
  121. self.b = tk.Button(self.frame,bg=bg,text="SAVE", width=5)#,command=self.event) #bv.change_dmx)
  122. self.b["command"] = self.save_fixture
  123. #self.b.pack( side=tk.LEFT)
  124. #self.b = tk.Button(self.frame,bg=bg,text="SAVE AS", width=5)#,command=self.event) #bv.change_dmx)
  125. #self.b["command"] = self.save_as_fixture
  126. #self.b.pack( side=tk.LEFT)
  127. self.b = tk.Label(self.frame,bg="black",text="") # spacer
  128. self.b.pack(fill=tk.Y, side=tk.LEFT)
  129. self.b = tk.Button(self.frame,bg="grey",text="HELP", width=5)#,command=self.event) #bv.change_dmx)
  130. self.b["command"] = libtk.online_help("librelight:80-fixture-editor" ) #""fixture-editor")
  131. self.b.pack( side=tk.LEFT)
  132. self.b = tk.Button(self.frame,bg="grey",text="VDIM", width=5)#,command=self.event) #bv.change_dmx)
  133. self.b["command"] = libtk.online_help("librelight:05-virtueller-attribute" ) #""fixture-editor")
  134. self.b.pack( side=tk.LEFT)
  135. # HEAD 2
  136. r=0
  137. c=0
  138. self.frame = tk.Frame(root,bg="grey",width=width)
  139. self.frame.pack(fill="both", side=tk.TOP)
  140. #c+=1 ;r=0
  141. self.b = tk.Label(self.frame,bg="#ddd",text="FIX-ID:")
  142. self.b.grid(row=r,column=c) #,expand=1)
  143. r+=1
  144. self.b = tk.Button(self.frame,bg="lightblue",text="3001", width=4)
  145. self.fixid=self.b
  146. self.b["command"] = self.set_fixid
  147. self.b.grid(row=r,column=c)
  148. c+=1 ;r=0
  149. self.b = tk.Label(self.frame,bg="#ddd",text="NAME:")
  150. self.b.grid(row=r,column=c) #,expand=1)
  151. r+=1
  152. self.b = tk.Button(self.frame,bg="lightblue",text="FixName", width=13)
  153. self.name=self.b
  154. self.b["command"] = self.set_name
  155. self.b.grid(row=r,column=c)
  156. c+=1 ;r=0
  157. self.b = tk.Label(self.frame,bg="lightblue",text="UNIV:")
  158. self.b.grid(row=r,column=c)
  159. r+=1
  160. self.b_univ = tk.Button(self.frame,bg="lightblue",text="1", width=4)#,command=self.event) #bv.change_dmx)
  161. self.entry_univ=self.b_univ
  162. self.b_univ["command"] = self.event_univ
  163. self.b_univ.grid(row=r,column=c)
  164. c+=1 ;r=0
  165. self.b = tk.Label(self.frame,bg="lightblue",text="DMX:")
  166. self.b.grid(row=r,column=c)
  167. r+=1
  168. self.b = tk.Button(self.frame,bg="lightblue",text="1", width=4)#,command=self.event) #bv.change_dmx)
  169. self.entry_dmx=self.b
  170. self.b["command"] = self.event_dmx
  171. self.b.grid(row=r,column=c)
  172. c+=1 ;r=0
  173. self.b = tk.Label(self.frame,bg="lightblue",text="QTY:")
  174. self.b.grid(row=r,column=c)
  175. r+=1
  176. self.b = tk.Button(self.frame,bg="lightblue",text="4", width=4)#,command=self.event) #bv.change_dmx)
  177. #self.entry_qty=self.b
  178. self.qty=self.b
  179. self.b["command"] = self.set_qty
  180. #self.b["command"] = self.event_dmx
  181. self.b.grid(row=r,column=c)
  182. c+=1 ;r=0
  183. r+=1
  184. self.b = tk.Button(self.frame,bg="lightblue",text="PATCH", width=6)#,command=self.event) #bv.change_dmx)
  185. self.b["command"] = self.do_patch
  186. self.b.grid(row=r,column=c)
  187. # HEAD 1
  188. #root = tk.Frame(root,bg="black",width=width)
  189. #root.pack(fill=tk.BOTH, side=tk.TOP)
  190. self.frame = tk.Frame(root,bg="grey",width=width)
  191. self.frame.pack(fill="x", side=tk.TOP)
  192. self.b = tk.Label(self.frame,bg="#fff",text="Fixture Editor") #,font=self.font8 )
  193. self.b.pack(fill=None, side=tk.LEFT)
  194. self.b = tk.Label(self.frame,bg="#aaa",text="FILE:") #,font=self.font8 )
  195. self.b.pack(fill=None, side=tk.LEFT)
  196. self.b_path = tk.Label(self.frame,bg="#fff",text="~/LibreLight/fixtures/lalla.json") #,font=self.font8 )
  197. self.b_path.pack(fill=None, side=tk.LEFT)
  198. self.b = tk.Label(self.frame,bg="#aaa",text="----") #,font=self.font8 )
  199. self.b.pack(fill=None, side=tk.LEFT)
  200. self.b_info = tk.Label(self.frame,bg="#fff",text="") #,font=self.font8 )
  201. self.b_info.pack(fill=None, side=tk.LEFT)
  202. # DATA
  203. self.frame = libtk.ScrollFrame(root,bg="#003",width=2000 ,height=1000,bd=2) # fader frame
  204. self.r=0
  205. self.c=0
  206. self.pb=10
  207. self.j = 0
  208. for page_nr in range(10):
  209. self.draw_bank(page_nr)
  210. self._event_redraw()
  211. def draw_bank(self,page_nr):
  212. title = self.title
  213. width = self.width
  214. idx = self.pb*page_nr
  215. data = self.data[idx:idx+self.pb]
  216. c = idx
  217. for j,row in enumerate(data):
  218. if c % self.pb == 0 or c==0:
  219. h=hex(j*10)[2:].rjust(2,"0")
  220. frameS = tk.Frame(self.frame,bg="#000",width=width,border=2)
  221. frameS.pack(fill=tk.BOTH, side=tk.TOP)
  222. bank_nr=j//self.pb+1
  223. txt="BANK:{} {}-{}".format(bank_nr,bank_nr*self.pb-self.pb+1,bank_nr*self.pb)
  224. self.b = tk.Label(frameS,bg="lightblue",text=txt,width=15,font=self.font8 )
  225. self.header.append(self.b)
  226. self.b.pack(fill=None, side=tk.LEFT)
  227. self.b = tk.Label(frameS,bg="black",text="" ,width=11,font=self.font8 )
  228. self.b.pack(fill=tk.BOTH, side=tk.LEFT)
  229. try:
  230. frameS = tk.Frame(self.frame,bg="#a000{}".format(h),width=width,border=2)
  231. except:
  232. frameS = tk.Frame(self.frame,bg="#a0aadd",width=width,border=2)
  233. self.c=0
  234. e= libtk.ELEM_FADER(frameS,nr=j+1,cb=self._cb,fader_cb=self._fader_cb,width=14)
  235. e.pack()
  236. #e.attr["bg"] = "red"
  237. self.fader_elem.append(e)
  238. self.elem.append(e)
  239. frameS.pack(fill=tk.X, side=tk.TOP)
  240. c+=1
  241. def _fader_cb(self,arg,name="<name>",**args):
  242. print(" FixtureEditor._cb",args,arg,name)
  243. #print(" ",name,"_cb.args >>",args,arg[1:])
  244. self.count_ch()
  245. try:
  246. a1 = arg #arg[2]
  247. nr = args["nr"] #.nr
  248. j=[]
  249. e = self.fader_elem[args["nr"]-1]
  250. nr = int(e.elem_nr["text"])
  251. if not nr:
  252. nr = args["nr"]
  253. return
  254. nr_start = ( int(self.entry_dmx["text"])-1 + int(self.entry_univ["text"])*512 )
  255. nr += nr_start
  256. jdata = {'VALUE': int(a1), 'args': [] , 'FADE': 0,'DMX': str(nr)}
  257. print(" ",jdata)
  258. j.append(jdata)
  259. MAIN.jclient_send(j)
  260. except Exception as e:
  261. print("exec",arg,args,nr)
  262. print(e)
  263. raise e
  264. def _cb(self,arg,name="<name>",**args):
  265. #print(" FixtureEditor._cb")
  266. #print(" ",name,"_cb.args >>",args,arg)
  267. self.count_ch()
  268. elm,mode,name = arg
  269. #print("_cb",arg)
  270. if mode == "set_nr":
  271. try:
  272. int(name)
  273. except Exception as e:
  274. print(self,"_cb",e)
  275. print(dir(elm),elm.elem_nr)
  276. elm._set_nr("Number!") #"ERROR") #nr["text"] = "ERR"
  277. if mode == "set_attr":
  278. #print(" ->")
  279. if name.endswith(" FINE"):
  280. name = name.replace(" FINE","-FINE")
  281. elm.label["text"] = name
  282. def count_ch(self):
  283. #print("FixtureEditor.count_ch:")
  284. #e._set_attr( "---")
  285. ch_s = []
  286. j=-1
  287. self.b_info["text"] = "xx"
  288. for i,elem in enumerate(self.elem):
  289. #print(dir(elem))
  290. txt = elem.attr["text"]
  291. if txt:
  292. #print("count_ch:",i,txt)
  293. if txt.startswith("EMPTY"):
  294. elem.attr["bg"] = "#fa0"
  295. elem.attr["activebackground"] = "#ff0"
  296. elif txt.endswith("-FINE"):
  297. elem.attr["bg"] = "#0b0"
  298. elem.attr["activebackground"] = "#ff0"
  299. else:
  300. elem.attr["bg"] = "#0f0"
  301. elem.attr["activebackground"] = "#ff0"
  302. ch_s.append([i,txt])
  303. j=i
  304. else:
  305. elem.attr["bg"] = "lightgrey"
  306. elem.attr["activebackground"] = "white"
  307. self.b_info["text"] = "CH's: {} USED: {}".format(j+1,len(ch_s))
  308. def set_qty(self,_event=None):
  309. txt = self.qty["text"]
  310. def _cb(data):
  311. if not data:
  312. print("err443",self,"_cb",data)
  313. return None
  314. txt = data["Value"]
  315. print(self,"._cb()",txt)
  316. self.qty["text"] = "{}".format(txt)
  317. print("set_qty",[_event,self])
  318. dialog._cb = _cb
  319. dialog.askstring("QTY:","QTY:",initialvalue=txt)
  320. def do_patch(self,_event=None):
  321. qty = int(self.qty["text"])
  322. ID = int(self.fixid["text"])
  323. univ = self.univ #int(self.univ["text"])
  324. dmx = self.dmx #int(self.dmx["text"])
  325. name = self.name["text"]
  326. name_nr = ""
  327. if "-" in name:
  328. try:
  329. name_nr = name.split("-")[-1]
  330. name_nr = int(name_nr)
  331. name = name.split("-")[:-1]
  332. name = "-".join(name)
  333. except:
  334. name_nr = ""
  335. if name_nr == '':
  336. if ID:
  337. name_nr = ID
  338. else:
  339. name_nr = 9000
  340. name_nr=int(name_nr)
  341. print("do_patch",dmx,univ,qty)
  342. DMX = dmx #univ*512 + dmx
  343. fixture = {"DMX": DMX, "UNIVERS": univ, "NAME": "D1", "TYPE": "", "VENDOR": "", "ATTRIBUT": {} , "ACTIVE": 0}
  344. ATTR = []
  345. max_nr = 0
  346. for fader in self.fader_elem:
  347. #print("patch -",fader)
  348. attr = fader.attr["text"]
  349. nr = fader.elem_nr["text"]
  350. if nr == '':
  351. continue
  352. if nr == "off":
  353. nr = -1
  354. try:
  355. nr = int(nr)
  356. except ValueError as e:
  357. _fader_nr = fader.label["text"].split()[0]
  358. _msg = " Channel "+str(_fader_nr)+"\n"
  359. #_msg += " VAL = "+str(nr)+"\n"
  360. _msg += "NR: Is not a Number !"
  361. r=tkinter.messagebox.showwarning(message=_msg,title="Patch Error",parent=None)
  362. return
  363. mode = fader.mode["text"]
  364. val = float(fader.elem_fader.get())
  365. if not attr:
  366. continue
  367. if attr.startswith("EMPTY"):
  368. continue
  369. ATTR = OrderedDict()
  370. ATTR["NR"] = nr
  371. ATTR["MASTER"] = "0"
  372. ATTR["MODE"] = mode
  373. ATTR["VALUE"] = val
  374. ATTR["ACTIVE"] = 0
  375. ATTR["FX"] = ""
  376. ATTR["FX2"] = {}
  377. print("patch --",nr,mode,attr)
  378. fixture["ATTRIBUT"][attr] = ATTR
  379. if nr > max_nr:
  380. max_nr = nr
  381. ok = 1
  382. out=[]
  383. err = []
  384. msg=""
  385. err2 = []
  386. sucess = []
  387. _fixture = fixture
  388. _dmx = _fixture["DMX"]
  389. for i in range(qty):
  390. fixture = copy.deepcopy(_fixture)
  391. fixture["DMX"] = _dmx
  392. print("i",i)
  393. fixture["NAME"] = name + "-{:0>4}".format(name_nr)
  394. fixture["ID"] = ID
  395. print(fixture)
  396. fixture = fixlib.FIXTURE_CHECK_SDATA(ID,fixture)
  397. #out.append(sdata)
  398. out.append(fixture)
  399. #fixture = copy.deepcopy(fixture)
  400. if str(ID) in MAIN.FIXTURES.fixtures:
  401. ok = 0
  402. #err.append(" ID '{}' is in use ! ".format(ID))
  403. err.append("FIX-ID '{}' ".format(ID))
  404. if ATTR:
  405. sucess.append("ID '{}' DMX:{} UNIV:{}".format(ID,fixture["DMX"],fixture["UNIVERS"]))
  406. else:
  407. ok = 0
  408. err2.append(" NO 'attributes' ID:'{}' ! ".format(ID))
  409. name_nr += 1
  410. ID += 1
  411. _dmx += max_nr # bug offset of one fixture
  412. print("OK:",ok)
  413. print()
  414. if err:
  415. #msg+="Name:'"+name+"'\n"
  416. msg+="FIX-ID is in use !\n"
  417. msg+="\n"
  418. msg+="\n".join(err)
  419. msg+="\n"
  420. msg+="\n"
  421. msg+="OVERWRITE ?\n"
  422. msg+="überschreiben ?\n"
  423. #msg+="\n "
  424. r=tkinter.messagebox.askyesno(message=msg,title="cancel/Abbruch",parent=None)
  425. print("err",r)
  426. if r: # if yes
  427. pass
  428. else:
  429. return
  430. if err2:
  431. r=tkinter.messagebox.showwarning(message="PATCH ERROR 2'"+name+"'\n\n"+"\n".join(err2)+"\n\n ",title="Error",parent=None)
  432. return
  433. if sucess:
  434. msg+="name:'"+name+"'\n\n"
  435. msg="PATCH OK ?\n"
  436. msg+="\n".join(sucess)
  437. msg+="\n"
  438. r=tkinter.messagebox.askyesno(message=msg,title="Execute/Ausführen",parent=None)
  439. print("yes no" ,r )
  440. if r:
  441. for fix in out:
  442. print(";;",fix)
  443. k = str(fix["ID"])
  444. v = fix
  445. MAIN.FIXTURES.fixtures[k] = v
  446. MAIN.FIXTURES._re_sort()
  447. def set_fixid(self,_event=None):
  448. txt = self.fixid["text"]
  449. def _cb(data):
  450. if not data:
  451. print("err443",self,"_cb",data)
  452. return None
  453. txt = data["Value"]
  454. print(self,"._cb()",txt)
  455. self.fixid["text"] = "{}".format(txt)
  456. print("set_fixid",[_event,self])
  457. dialog._cb = _cb
  458. dialog.askstring("FIXTURE ID:","ID:",initialvalue=txt)
  459. def set_name(self,_event=None):
  460. txt = self.name["text"]
  461. def _cb(data):
  462. if not data:
  463. print("err443",self,"_cb",data)
  464. return None
  465. txt = data["Value"]
  466. print(self,"._cb()",txt)
  467. self.name["text"] = "{}".format(txt)
  468. print("change_dmx",[_event,self])
  469. dialog._cb = _cb
  470. dialog.askstring("FIXTURE NAME:","NAME:",initialvalue=txt)
  471. def save_as_fixture(self,event=None):
  472. print("save_as_fix",self,event)
  473. self.count_ch()
  474. def save_fixture(self,event=None):
  475. print("save_fix",self,event)
  476. self.count_ch()
  477. def open_fixture_list_global(self):
  478. self.close_fixture_list()
  479. self._open_fixture_list(mode="GLOBAL")
  480. def open_fixture_list_import(self):
  481. self.close_fixture_list()
  482. self._open_fixture_list(mode="IMPORT")
  483. def open_fixture_list_user(self):
  484. self.close_fixture_list()
  485. self._open_fixture_list(mode="USER")
  486. def _open_fixture_list(self,mode=""):
  487. name = "FIXTURE-{}".format(mode)
  488. line1="Fixture Library"
  489. line2="CHOOS to EDIT >> DEMO MODUS"
  490. line3="CHOOS to EDIT >> DEMO MODUS"
  491. cb = None #LOAD_FIXTURE(self,"USER").cb
  492. self.pw = libtk.PopupList(name,width=600,cb=cb,left=libtk._POS_LEFT+620,bg="#333")
  493. frame = self.pw.sframe(line1=line1,line2=line2) #,line3=line3)
  494. def cb(event=None,args={}):
  495. print("open_fixture_list.cb(")
  496. print(" ",args)
  497. if self.pw:
  498. self.pw.w.tk.destroy()
  499. data = args["data"]
  500. self.b_path["text"] = data["name"] #"load_MH2"
  501. self.name["text"] = data["name"] #"load_MH2"
  502. self.name["text"] = data["name"] #"load_MH2"
  503. xpath = data["xpath"] + "/" + data["xfname"]
  504. fdata = showlib._read_sav_file(xpath)
  505. n = []
  506. a = []
  507. m = []
  508. NR = 0
  509. for row in fdata:
  510. #print("row: ",row.keys())
  511. #print("a-")
  512. for k,fixture in row.items():#keys():
  513. #v = row[k]
  514. if "NAME" not in fixture:
  515. continue
  516. if fixture["NAME"] != args["val"]:
  517. continue
  518. self.fixture = fixture
  519. print("a :",k,str(fixture)[:220],"...")
  520. #print("a ::",type(k),":",type(fixture))
  521. attr_by_nr = fixlib.fixture_order_attr_by_nr(fixture)
  522. if "ATTRIBUT" in fixture:
  523. for at in attr_by_nr: #fixture["ATTRIBUT"]:
  524. print(" ",at)
  525. if at.startswith("EMPTY"):
  526. NR += 1
  527. n.append(str(NR))
  528. a.append(at)
  529. m.append("-")
  530. continue
  531. print(" ",fixture["ATTRIBUT"][at])
  532. if at.startswith("_"):
  533. continue
  534. try:
  535. NR = fixture["ATTRIBUT"][at]["NR"]
  536. n.append(str(NR))
  537. except:
  538. n.append("-")
  539. at2 = at
  540. if at2.endswith(" FINE"):
  541. at2 = at2.replace(" FINE","-FINE")
  542. while " -" in at2:
  543. at2 = at2.replace(" -","-")
  544. at2 = at2.replace(" "," ")
  545. a.append(at2 ) #+":"+ str(fixture["ATTRIBUT"][at]["NR"]))
  546. if at.endswith("-FINE"):
  547. m.append("-")
  548. elif at in MAIN._FIX_FADE_ATTR:
  549. #["PAN","TILT","DIM","RED","GREEN","BLUE","CYAN","YELLOW","MAGENTA","FOCUS","ZOOM","FROST"]:
  550. m.append("F")
  551. else:
  552. m.append("S")
  553. #m.append("F")
  554. break # only a single fixture #no sub fixture
  555. self._load_fix(None,n,a,m)
  556. self.close_fixture_list()
  557. #_x =dir(MAIN)
  558. #_x.sort()
  559. #for _a in _x:
  560. # print(_a)
  561. blist = fixlib._load_fixture_list(mode=mode)
  562. r=GUI_LOAD_FIXTURE_LIST(frame,data=blist,cb=cb,bg="#333")
  563. def close_fixture_list(self):
  564. if self.pw:
  565. self.pw.w.tk.destroy()
  566. def clear(self,_event=None,attr=[]):
  567. attr = [""]*100
  568. mode = [""]*100
  569. self._load_fix(None,attr,mode)
  570. self.b_path["text"] = "clean..."
  571. self.close_fixture_list()
  572. def load(self,_event=None,attr=[]):
  573. attr = ["LOAD"]*10
  574. mode = ["X"]*10
  575. self._load_fix(None,attr,mode)
  576. self.b_path["text"] = "clean..."
  577. self.close_fixture_list()
  578. def load_EMPTY(self,_event=None,attr=[]):
  579. #attr = [,"RED","GREEN","BLUE"]
  580. #mode = ["F","F","F","F"]
  581. self._load_mh(None)#,attr,mode)
  582. def load_DIM(self,_event=None,attr=[]):
  583. attr = ["DIM"]
  584. mode = ["F"]
  585. self._load_fix(None,attr,mode)
  586. self.b_path["text"] = "load_DIM"
  587. self.close_fixture_list()
  588. def load_LED(self,_event=None,attr=[]):
  589. attr = ["DIM","RED","GREEN","BLUE"]
  590. mode = ["F","F","F","F"]
  591. self._load_fix(None,attr,mode)
  592. self.b_path["text"] = "load_LED"
  593. self.close_fixture_list()
  594. def load_MH(self,_event=None,attr=[]):
  595. if not attr:
  596. attr = ["PAN","PAN-FINE","TILT","TILT-FINE","SHUTTER","DIM","RED","GREEN","BLUE","GOBO"]
  597. mode = []
  598. for a in attr:
  599. if a.endswith("-FINE"):
  600. mode.append("-")
  601. elif a in MAIN._FIX_FADE_ATTR: #["PAN","TILT","DIM","RED","GREEN","BLUE","CYAN","YELLOW","MAGENTA","FOCUS","ZOOM","FROST"]:
  602. mode.append("F")
  603. else:
  604. mode.append("S")
  605. self._load_fix(None,attr,mode)
  606. self.b_path["text"] = "load_MH"
  607. self.close_fixture_list()
  608. def load_MH2(self,_event=None,attr=[]):
  609. attr = ["PAN","PAN-FINE","TILT","TILT-FINE","SHUTTER","DIM","RED","GREEN","BLUE","GOBO","G-ROT","PRISM","P-ROT","ZOOM","CONTR"]
  610. mode = ["F","F","F","F","S","F","F","F","F","S","S","S","S","F","S"]
  611. self.b_path["text"] = "load_MH2"
  612. self._load_fix(None,attr,mode)
  613. self.close_fixture_list()
  614. def _load_fix(self,_event=None,nrs=[],attr=[],mode=[]):
  615. print("load_fixture",[_event,self])
  616. #for i,e in enumerate(self.elem):
  617. for i,e in enumerate(self.elem):
  618. #print(self,"event",_event,e)
  619. #print("event",_event,e)
  620. if len(attr) > i:
  621. e._set_nr( nrs[i])
  622. else:
  623. e._set_nr( "")
  624. if len(attr) > i:
  625. e._set_attr( attr[i])
  626. else:
  627. e._set_attr( "")
  628. if len(mode) > i:
  629. e._set_mode( mode[i]+"'")
  630. else:
  631. e._set_mode( "---")
  632. self.count_ch()
  633. def event_univ(self,_event=None):
  634. nr=self.univ
  635. txt= self.entry_univ["text"]
  636. #def _cb(txt):
  637. def _cb(data):
  638. if not data:
  639. print("err443",self,"_cb",data)
  640. return None
  641. txt = data["Value"]
  642. print(self,"event_univ._cb()",txt)
  643. try:
  644. nr = int(txt)
  645. except TypeError:
  646. print("--- abort ---")
  647. return 0
  648. self.univ = nr
  649. self._event_redraw(_event)
  650. dialog._cb = _cb
  651. dialog.askstring("Universe","Univ 0-15",initialvalue=txt)
  652. def event_dmx(self,_event=None):
  653. nr=self.dmx
  654. txt= self.entry_dmx["text"]
  655. #txt = dialog.askstring("DMX","ArtNet 1-512 (7680 max)",initialvalue=txt)
  656. #def _cb(txt):
  657. def _cb(data):
  658. if not data:
  659. print("err443",self,"_cb",data)
  660. return None
  661. txt = data["Value"]
  662. print(self,"event_dmx._cb()",txt)
  663. try:
  664. nr = int(txt)
  665. except TypeError:
  666. print("--- abort ---")
  667. return 0
  668. self.dmx = nr
  669. if self.dmx <= 0:
  670. self.dmx = 1
  671. if self.dmx > 512:
  672. self.univ = (self.dmx-1)//512
  673. self.dmx = (self.dmx-1)%512+1
  674. self._event_redraw(_event)
  675. dialog._cb = _cb
  676. dialog.askstring("DMX","ArtNet 1-512 (7680 max)",initialvalue=txt)
  677. def _event_redraw(self,_event=None):
  678. self.entry_dmx["text"] = "{}".format(self.dmx)
  679. self.entry_univ["text"] = "{}".format(self.univ)
  680. nr = 1 # self.univ*(512)+self.dmx # multiplay fader nr with dmx...
  681. # self.b_xdmx["text"] = " {} ".format(nr)
  682. print("change_dmx",[_event,self])
  683. for i,btn in enumerate(self.elem):
  684. #print("event",_event,btn)
  685. #print("btn",btn)
  686. dmx=nr+i
  687. nr2 = dmx%512
  688. btn.set_label("{} {}.{}\n D:{}".format(i+1,self.univ,nr2,dmx))
  689. btn.nr = nr+i
  690. pb=self.pb
  691. for j,e in enumerate(self.header):
  692. p=j+1
  693. #p=nr/pb
  694. txt="BANK:{} {}-{}".format(p,p*pb-pb+nr,(p*pb+nr)-1)
  695. print("---",j,txt,e)
  696. e["text"] = txt
  697. self.count_ch()