fix.py 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791
  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=12
  207. self.j = 0
  208. for page_nr in range(12):
  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)
  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[1:])
  267. self.count_ch()
  268. def count_ch(self):
  269. #print("FixtureEditor.count_ch:")
  270. #e._set_attr( "---")
  271. ch_s = []
  272. j=-1
  273. self.b_info["text"] = "xx"
  274. for i,elem in enumerate(self.elem):
  275. #print(dir(elem))
  276. txt = elem.attr["text"]
  277. if txt:
  278. #print("count_ch:",i,txt)
  279. if txt.startswith("EMPTY"):
  280. elem.attr["bg"] = "#fa0"
  281. elem.attr["activebackground"] = "#fa0"
  282. else:
  283. elem.attr["bg"] = "#0f0"
  284. elem.attr["activebackground"] = "#0fa"
  285. ch_s.append([i,txt])
  286. j=i
  287. else:
  288. elem.attr["bg"] = "lightgrey"
  289. elem.attr["activebackground"] = "white"
  290. self.b_info["text"] = "CH's: {} USED: {}".format(j+1,len(ch_s))
  291. def set_qty(self,_event=None):
  292. txt = self.qty["text"]
  293. def _cb(data):
  294. if not data:
  295. print("err443",self,"_cb",data)
  296. return None
  297. txt = data["Value"]
  298. print(self,"._cb()",txt)
  299. self.qty["text"] = "{}".format(txt)
  300. print("set_qty",[_event,self])
  301. dialog._cb = _cb
  302. dialog.askstring("QTY:","QTY:",initialvalue=txt)
  303. def do_patch(self,_event=None):
  304. qty = int(self.qty["text"])
  305. ID = int(self.fixid["text"])
  306. univ = self.univ #int(self.univ["text"])
  307. dmx = self.dmx #int(self.dmx["text"])
  308. name = self.name["text"]
  309. name_nr = ""
  310. if "-" in name:
  311. try:
  312. name_nr = name.split("-")[-1]
  313. name_nr = int(name_nr)
  314. name = name.split("-")[:-1]
  315. name = "-".join(name)
  316. except:
  317. name_nr = ""
  318. if name_nr == '':
  319. if ID:
  320. name_nr = ID
  321. else:
  322. name_nr = 9000
  323. name_nr=int(name_nr)
  324. print("do_patch",dmx,univ,qty)
  325. DMX = dmx #univ*512 + dmx
  326. fixture = {"DMX": DMX, "UNIVERS": univ, "NAME": "D1", "TYPE": "", "VENDOR": "", "ATTRIBUT": {} , "ACTIVE": 0}
  327. ATTR = []
  328. max_nr = 0
  329. for fader in self.fader_elem:
  330. #print("patch -",fader)
  331. attr = fader.attr["text"]
  332. nr = fader.elem_nr["text"]
  333. if nr == '':
  334. continue
  335. if nr == "off":
  336. nr = -1
  337. nr = int(nr)
  338. mode = fader.mode["text"]
  339. val = float(fader.elem_fader.get())
  340. if not attr:
  341. continue
  342. if attr.startswith("EMPTY"):
  343. continue
  344. ATTR = OrderedDict()
  345. ATTR["NR"] = nr
  346. ATTR["MASTER"] = "0"
  347. ATTR["MODE"] = mode
  348. ATTR["VALUE"] = val
  349. ATTR["ACTIVE"] = 0
  350. ATTR["FX"] = ""
  351. ATTR["FX2"] = {}
  352. print("patch --",nr,mode,attr)
  353. fixture["ATTRIBUT"][attr] = ATTR
  354. if nr > max_nr:
  355. max_nr = nr
  356. ok = 1
  357. out=[]
  358. err = []
  359. msg=""
  360. err2 = []
  361. sucess = []
  362. _fixture = fixture
  363. _dmx = _fixture["DMX"]
  364. for i in range(qty):
  365. fixture = copy.deepcopy(_fixture)
  366. fixture["DMX"] = _dmx
  367. print("i",i)
  368. fixture["NAME"] = name + "-{:0>4}".format(name_nr)
  369. fixture["ID"] = ID
  370. print(fixture)
  371. fixture = fixlib.FIXTURE_CHECK_SDATA(ID,fixture)
  372. #out.append(sdata)
  373. out.append(fixture)
  374. #fixture = copy.deepcopy(fixture)
  375. if str(ID) in MAIN.FIXTURES.fixtures:
  376. ok = 0
  377. #err.append(" ID '{}' is in use ! ".format(ID))
  378. err.append("FIX-ID '{}' ".format(ID))
  379. if ATTR:
  380. sucess.append("ID '{}' DMX:{} UNIV:{}".format(ID,fixture["DMX"],fixture["UNIVERS"]))
  381. else:
  382. ok = 0
  383. err2.append(" NO 'attributes' ID:'{}' ! ".format(ID))
  384. name_nr += 1
  385. ID += 1
  386. _dmx += max_nr # bug offset of one fixture
  387. print("OK:",ok)
  388. print()
  389. if err:
  390. #msg+="Name:'"+name+"'\n"
  391. msg+="FIX-ID is in use !\n"
  392. msg+="\n"
  393. msg+="\n".join(err)
  394. msg+="\n"
  395. msg+="\n"
  396. msg+="OVERWRITE ?\n"
  397. msg+="überschreiben ?\n"
  398. #msg+="\n "
  399. r=tkinter.messagebox.askyesno(message=msg,title="cancel/Abbruch",parent=None)
  400. print("err",r)
  401. if r: # if yes
  402. pass
  403. else:
  404. return
  405. if err2:
  406. r=tkinter.messagebox.showwarning(message="PATCH ERROR 2'"+name+"'\n\n"+"\n".join(err2)+"\n\n ",title="Error",parent=None)
  407. return
  408. if sucess:
  409. msg+="name:'"+name+"'\n\n"
  410. msg="PATCH OK ?\n"
  411. msg+="\n".join(sucess)
  412. msg+="\n"
  413. r=tkinter.messagebox.askyesno(message=msg,title="Execute/Ausführen",parent=None)
  414. print("yes no" ,r )
  415. if r:
  416. for fix in out:
  417. print(";;",fix)
  418. k = str(fix["ID"])
  419. v = fix
  420. MAIN.FIXTURES.fixtures[k] = v
  421. MAIN.FIXTURES._re_sort()
  422. def set_fixid(self,_event=None):
  423. txt = self.fixid["text"]
  424. def _cb(data):
  425. if not data:
  426. print("err443",self,"_cb",data)
  427. return None
  428. txt = data["Value"]
  429. print(self,"._cb()",txt)
  430. self.fixid["text"] = "{}".format(txt)
  431. print("set_fixid",[_event,self])
  432. dialog._cb = _cb
  433. dialog.askstring("FIXTURE ID:","ID:",initialvalue=txt)
  434. def set_name(self,_event=None):
  435. txt = self.name["text"]
  436. def _cb(data):
  437. if not data:
  438. print("err443",self,"_cb",data)
  439. return None
  440. txt = data["Value"]
  441. print(self,"._cb()",txt)
  442. self.name["text"] = "{}".format(txt)
  443. print("change_dmx",[_event,self])
  444. dialog._cb = _cb
  445. dialog.askstring("FIXTURE NAME:","NAME:",initialvalue=txt)
  446. def save_as_fixture(self,event=None):
  447. print("save_as_fix",self,event)
  448. self.count_ch()
  449. def save_fixture(self,event=None):
  450. print("save_fix",self,event)
  451. self.count_ch()
  452. def open_fixture_list_global(self):
  453. self.close_fixture_list()
  454. self._open_fixture_list(mode="GLOBAL")
  455. def open_fixture_list_import(self):
  456. self.close_fixture_list()
  457. self._open_fixture_list(mode="IMPORT")
  458. def open_fixture_list_user(self):
  459. self.close_fixture_list()
  460. self._open_fixture_list(mode="USER")
  461. def _open_fixture_list(self,mode=""):
  462. name = "FIXTURE-{}".format(mode)
  463. line1="Fixture Library"
  464. line2="CHOOS to EDIT >> DEMO MODUS"
  465. line3="CHOOS to EDIT >> DEMO MODUS"
  466. cb = None #LOAD_FIXTURE(self,"USER").cb
  467. self.pw = libtk.PopupList(name,width=600,cb=cb,left=libtk._POS_LEFT+620,bg="#333")
  468. frame = self.pw.sframe(line1=line1,line2=line2) #,line3=line3)
  469. def cb(event=None,args={}):
  470. print("open_fixture_list.cb(")
  471. print(" ",args)
  472. if self.pw:
  473. self.pw.w.tk.destroy()
  474. data = args["data"]
  475. self.b_path["text"] = data["name"] #"load_MH2"
  476. self.name["text"] = data["name"] #"load_MH2"
  477. self.name["text"] = data["name"] #"load_MH2"
  478. xpath = data["xpath"] + "/" + data["xfname"]
  479. fdata = showlib._read_sav_file(xpath)
  480. n = []
  481. a = []
  482. m = []
  483. NR = 0
  484. for row in fdata:
  485. #print("row: ",row.keys())
  486. #print("a-")
  487. for k,fixture in row.items():#keys():
  488. #v = row[k]
  489. if "NAME" not in fixture:
  490. continue
  491. if fixture["NAME"] != args["val"]:
  492. continue
  493. self.fixture = fixture
  494. print("a :",k,str(fixture)[:220],"...")
  495. #print("a ::",type(k),":",type(fixture))
  496. attr_by_nr = fixlib.fixture_order_attr_by_nr(fixture)
  497. if "ATTRIBUT" in fixture:
  498. for at in attr_by_nr: #fixture["ATTRIBUT"]:
  499. print(" ",at)
  500. if at.startswith("EMPTY"):
  501. NR += 1
  502. n.append(str(NR))
  503. a.append(at)
  504. m.append("-")
  505. continue
  506. print(" ",fixture["ATTRIBUT"][at])
  507. if at.startswith("_"):
  508. continue
  509. try:
  510. NR = fixture["ATTRIBUT"][at]["NR"]
  511. n.append(str(NR))
  512. except:
  513. n.append("-")
  514. a.append(at ) #+":"+ str(fixture["ATTRIBUT"][at]["NR"]))
  515. if at.endswith("-FINE"):
  516. m.append("-")
  517. elif at in MAIN._FIX_FADE_ATTR: #["PAN","TILT","DIM","RED","GREEN","BLUE","CYAN","YELLOW","MAGENTA","FOCUS","ZOOM","FROST"]:
  518. m.append("F")
  519. else:
  520. m.append("S")
  521. #m.append("F")
  522. break # only a single fixture #no sub fixture
  523. self._load_fix(None,n,a,m)
  524. self.close_fixture_list()
  525. #_x =dir(MAIN)
  526. #_x.sort()
  527. #for _a in _x:
  528. # print(_a)
  529. blist = fixlib._load_fixture_list(mode=mode)
  530. r=GUI_LOAD_FIXTURE_LIST(frame,data=blist,cb=cb,bg="#333")
  531. def close_fixture_list(self):
  532. if self.pw:
  533. self.pw.w.tk.destroy()
  534. def clear(self,_event=None,attr=[]):
  535. attr = [""]*100
  536. mode = [""]*100
  537. self._load_fix(None,attr,mode)
  538. self.b_path["text"] = "clean..."
  539. self.close_fixture_list()
  540. def load(self,_event=None,attr=[]):
  541. attr = ["LOAD"]*10
  542. mode = ["X"]*10
  543. self._load_fix(None,attr,mode)
  544. self.b_path["text"] = "clean..."
  545. self.close_fixture_list()
  546. def load_EMPTY(self,_event=None,attr=[]):
  547. #attr = [,"RED","GREEN","BLUE"]
  548. #mode = ["F","F","F","F"]
  549. self._load_mh(None)#,attr,mode)
  550. def load_DIM(self,_event=None,attr=[]):
  551. attr = ["DIM"]
  552. mode = ["F"]
  553. self._load_fix(None,attr,mode)
  554. self.b_path["text"] = "load_DIM"
  555. self.close_fixture_list()
  556. def load_LED(self,_event=None,attr=[]):
  557. attr = ["DIM","RED","GREEN","BLUE"]
  558. mode = ["F","F","F","F"]
  559. self._load_fix(None,attr,mode)
  560. self.b_path["text"] = "load_LED"
  561. self.close_fixture_list()
  562. def load_MH(self,_event=None,attr=[]):
  563. if not attr:
  564. attr = ["PAN","PAN-FINE","TILT","TILT-FINE","SHUTTER","DIM","RED","GREEN","BLUE","GOBO"]
  565. mode = []
  566. for a in attr:
  567. if a.endswith("-FINE"):
  568. mode.append("-")
  569. elif a in MAIN._FIX_FADE_ATTR: #["PAN","TILT","DIM","RED","GREEN","BLUE","CYAN","YELLOW","MAGENTA","FOCUS","ZOOM","FROST"]:
  570. mode.append("F")
  571. else:
  572. mode.append("S")
  573. self._load_fix(None,attr,mode)
  574. self.b_path["text"] = "load_MH"
  575. self.close_fixture_list()
  576. def load_MH2(self,_event=None,attr=[]):
  577. attr = ["PAN","PAN-FINE","TILT","TILT-FINE","SHUTTER","DIM","RED","GREEN","BLUE","GOBO","G-ROT","PRISM","P-ROT","ZOOM","CONTR"]
  578. mode = ["F","F","F","F","S","F","F","F","F","S","S","S","S","F","S"]
  579. self.b_path["text"] = "load_MH2"
  580. self._load_fix(None,attr,mode)
  581. self.close_fixture_list()
  582. def _load_fix(self,_event=None,nrs=[],attr=[],mode=[]):
  583. print("load_fixture",[_event,self])
  584. #for i,e in enumerate(self.elem):
  585. for i,e in enumerate(self.elem):
  586. #print(self,"event",_event,e)
  587. #print("event",_event,e)
  588. if len(attr) > i:
  589. e._set_nr( nrs[i])
  590. else:
  591. e._set_nr( "")
  592. if len(attr) > i:
  593. e._set_attr( attr[i])
  594. else:
  595. e._set_attr( "")
  596. if len(mode) > i:
  597. e._set_mode( mode[i]+"'")
  598. else:
  599. e._set_mode( "---")
  600. self.count_ch()
  601. def event_univ(self,_event=None):
  602. nr=self.univ
  603. txt= self.entry_univ["text"]
  604. #def _cb(txt):
  605. def _cb(data):
  606. if not data:
  607. print("err443",self,"_cb",data)
  608. return None
  609. txt = data["Value"]
  610. print(self,"event_univ._cb()",txt)
  611. try:
  612. nr = int(txt)
  613. except TypeError:
  614. print("--- abort ---")
  615. return 0
  616. self.univ = nr
  617. self._event_redraw(_event)
  618. dialog._cb = _cb
  619. dialog.askstring("Universe","Univ 0-15",initialvalue=txt)
  620. def event_dmx(self,_event=None):
  621. nr=self.dmx
  622. txt= self.entry_dmx["text"]
  623. #txt = dialog.askstring("DMX","ArtNet 1-512 (7680 max)",initialvalue=txt)
  624. #def _cb(txt):
  625. def _cb(data):
  626. if not data:
  627. print("err443",self,"_cb",data)
  628. return None
  629. txt = data["Value"]
  630. print(self,"event_dmx._cb()",txt)
  631. try:
  632. nr = int(txt)
  633. except TypeError:
  634. print("--- abort ---")
  635. return 0
  636. self.dmx = nr
  637. if self.dmx <= 0:
  638. self.dmx = 1
  639. if self.dmx > 512:
  640. self.univ = (self.dmx-1)//512
  641. self.dmx = (self.dmx-1)%512+1
  642. self._event_redraw(_event)
  643. dialog._cb = _cb
  644. dialog.askstring("DMX","ArtNet 1-512 (7680 max)",initialvalue=txt)
  645. def _event_redraw(self,_event=None):
  646. self.entry_dmx["text"] = "{}".format(self.dmx)
  647. self.entry_univ["text"] = "{}".format(self.univ)
  648. nr = 1 # self.univ*(512)+self.dmx # multiplay fader nr with dmx...
  649. # self.b_xdmx["text"] = " {} ".format(nr)
  650. print("change_dmx",[_event,self])
  651. for i,btn in enumerate(self.elem):
  652. #print("event",_event,btn)
  653. #print("btn",btn)
  654. dmx=nr+i
  655. nr2 = dmx%512
  656. btn.set_label("{} {}.{}\n D:{}".format(i+1,self.univ,nr2,dmx))
  657. btn.nr = nr+i
  658. pb=self.pb
  659. for j,e in enumerate(self.header):
  660. p=j+1
  661. #p=nr/pb
  662. txt="BANK:{} {}-{}".format(p,p*pb-pb+nr,(p*pb+nr)-1)
  663. print("---",j,txt,e)
  664. e["text"] = txt
  665. self.count_ch()