fix.py 28 KB

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