0
|
1 from __future__ import nested_scopes
|
136
|
2 import sys, time
|
0
|
3 sys.path.append('..')
|
|
4 from Widgets.Fadable import Fadable
|
|
5
|
|
6 from Tix import *
|
136
|
7 import math, atexit, pickle
|
|
8 from Submaster import Submasters, sub_maxes
|
|
9 import dmxclient
|
|
10 from uihelpers import toplevelat
|
0
|
11
|
|
12 nudge_keys = {
|
|
13 'up' : list('qwertyuiop'),
|
|
14 'down' : list('asdfghjkl')
|
|
15 }
|
|
16 nudge_keys['down'].append('semicolon')
|
|
17
|
|
18 class SubScale(Scale, Fadable):
|
|
19 def __init__(self, master, *args, **kw):
|
|
20 self.scale_var = kw.get('variable') or DoubleVar()
|
|
21 kw.update({'variable' : self.scale_var,
|
136
|
22 'from' : 1, 'to' : 0, 'showvalue' : 0,
|
146
|
23 'sliderlength' : 15, 'res' : 0.01,
|
|
24 'width' : 40, 'troughcolor' : 'black', 'bg' : 'grey40',
|
|
25 'highlightthickness' : 1, 'bd' : 1,
|
|
26 'highlightcolor' : 'red', 'highlightbackground' : 'black',
|
|
27 'activebackground' : 'red'})
|
0
|
28 Scale.__init__(self, master, *args, **kw)
|
136
|
29 Fadable.__init__(self, var=self.scale_var, wheel_step=0.05)
|
146
|
30 self.draw_indicator_colors()
|
|
31 def draw_indicator_colors(self):
|
|
32 if self.scale_var.get() == 0:
|
|
33 self['troughcolor'] = 'black'
|
|
34 else:
|
|
35 self['troughcolor'] = 'blue'
|
0
|
36
|
|
37 class SubmasterTk(Frame):
|
|
38 def __init__(self, master, name, current_level):
|
146
|
39 Frame.__init__(self, master, bd=1, relief='raised', bg='black')
|
0
|
40 self.slider_var = DoubleVar()
|
|
41 self.slider_var.set(current_level)
|
|
42 self.scale = SubScale(self, variable=self.slider_var, width=20)
|
146
|
43 namelabel = Label(self, text=name, font="Arial 8", bg='black',
|
|
44 fg='white')
|
|
45 namelabel.pack(side=TOP)
|
|
46 levellabel = Label(self, textvariable=self.slider_var, font="Arial 8",
|
|
47 bg='black', fg='white')
|
|
48 levellabel.pack(side=TOP)
|
0
|
49 self.scale.pack(side=BOTTOM, expand=1, fill=BOTH)
|
|
50
|
|
51 class KeyboardComposer(Frame):
|
148
|
52 def __init__(self, root, submasters, current_sub_levels=None, dmxdummy=0):
|
146
|
53 Frame.__init__(self, root, bg='black')
|
0
|
54 self.submasters = submasters
|
148
|
55 self.dmxdummy = dmxdummy
|
|
56
|
136
|
57 self.current_sub_levels = {}
|
|
58 if current_sub_levels:
|
|
59 self.current_sub_levels = current_sub_levels
|
|
60 else:
|
|
61 try:
|
|
62 self.current_sub_levels = \
|
|
63 pickle.load(file('.keyboardcomposer.savedlevels'))
|
|
64 except IOError:
|
|
65 pass
|
0
|
66
|
136
|
67 self.draw_ui()
|
|
68 self.send_levels_loop()
|
|
69 def draw_ui(self):
|
0
|
70 self.rows = [] # this holds Tk Frames for each row
|
|
71 self.slider_vars = {} # this holds subname:sub Tk vars
|
|
72 self.slider_table = {} # this holds coords:sub Tk vars
|
|
73 self.current_row = 0
|
|
74
|
|
75 self.make_key_hints()
|
|
76 self.draw_sliders()
|
|
77 self.highlight_row(self.current_row)
|
|
78 self.rows[self.current_row].focus()
|
136
|
79
|
146
|
80 self.buttonframe = Frame(self, bg='black')
|
|
81 self.buttonframe.pack(side=BOTTOM)
|
|
82 self.refreshbutton = Button(self.buttonframe, text="Refresh",
|
|
83 command=self.refresh, bg='black', fg='white')
|
|
84 self.refreshbutton.pack(side=LEFT)
|
|
85 self.save_stage_button = Button(self.buttonframe, text="Save",
|
|
86 command=lambda: self.save_current_stage(self.sub_name.get()),
|
|
87 bg='black', fg='white')
|
|
88 self.save_stage_button.pack(side=LEFT)
|
|
89 self.sub_name = Entry(self.buttonframe, bg='black', fg='white')
|
|
90 self.sub_name.pack(side=LEFT)
|
136
|
91 self.stop_frequent_update_time = 0
|
0
|
92 def make_key_hints(self):
|
|
93 keyhintrow = Frame(self)
|
|
94
|
|
95 col = 0
|
|
96 for upkey, downkey in zip(nudge_keys['up'],
|
|
97 nudge_keys['down']):
|
|
98 # what a hack!
|
|
99 downkey = downkey.replace('semicolon', ';')
|
136
|
100 upkey, downkey = (upkey.upper(), downkey.upper())
|
0
|
101
|
|
102 # another what a hack!
|
|
103 keylabel = Label(keyhintrow, text='%s\n%s' % (upkey, downkey),
|
146
|
104 width=1, font=('Arial', 10), bg='red', fg='white', anchor='c')
|
0
|
105 keylabel.pack(side=LEFT, expand=1, fill=X)
|
|
106 col += 1
|
|
107
|
|
108 keyhintrow.pack(fill=X, expand=0)
|
|
109 self.keyhints = keyhintrow
|
|
110 def setup_key_nudgers(self, tkobject):
|
|
111 for d, keys in nudge_keys.items():
|
|
112 for key in keys:
|
|
113 # lowercase makes full=0
|
|
114 keysym = "<KeyPress-%s>" % key
|
|
115 tkobject.bind(keysym, \
|
|
116 lambda evt, num=keys.index(key), d=d: \
|
|
117 self.got_nudger(num, d))
|
|
118
|
|
119 # uppercase makes full=1
|
|
120 keysym = "<KeyPress-%s>" % key.upper()
|
|
121 keysym = keysym.replace('SEMICOLON', 'colon')
|
|
122 tkobject.bind(keysym, \
|
|
123 lambda evt, num=keys.index(key), d=d: \
|
|
124 self.got_nudger(num, d, full=1))
|
|
125
|
|
126 # page up and page down change the row
|
|
127 for key in '<Prior> <Next> <Control-n> <Control-p>'.split():
|
|
128 tkobject.bind(key, self.change_row)
|
|
129
|
|
130 def change_row(self, event):
|
|
131 diff = 1
|
146
|
132 if event.keysym in ('Prior', 'p'):
|
0
|
133 diff = -1
|
|
134 old_row = self.current_row
|
|
135 self.current_row += diff
|
|
136 self.current_row = max(0, self.current_row)
|
|
137 self.current_row = min(len(self.rows) - 1, self.current_row)
|
|
138 self.unhighlight_row(old_row)
|
|
139 self.highlight_row(self.current_row)
|
|
140 row = self.rows[self.current_row]
|
|
141 self.keyhints.pack_configure(before=row)
|
|
142 def got_nudger(self, number, direction, full=0):
|
|
143 subtk = self.slider_table[(self.current_row, number)]
|
|
144 if direction == 'up':
|
|
145 if full:
|
136
|
146 subtk.scale.fade(1)
|
0
|
147 else:
|
|
148 subtk.scale.increase()
|
|
149 else:
|
|
150 if full:
|
|
151 subtk.scale.fade(0)
|
|
152 else:
|
|
153 subtk.scale.decrease()
|
|
154 def draw_sliders(self):
|
|
155 self.tk_focusFollowsMouse()
|
|
156
|
|
157 rowcount = -1
|
|
158 col = 0
|
136
|
159 for sub in self.submasters.get_all_subs():
|
0
|
160 if col == 0: # make new row
|
|
161 row = self.make_row()
|
|
162 rowcount += 1
|
|
163 current_level = self.current_sub_levels.get(sub.name, 0)
|
|
164 subtk = self.draw_sub_slider(row, col, sub.name, current_level)
|
|
165 self.slider_table[(rowcount, col)] = subtk
|
|
166 col += 1
|
|
167 col %= 10
|
136
|
168
|
146
|
169 def slider_changed(x, y, z, subtk=subtk):
|
|
170 subtk.scale.draw_indicator_colors()
|
|
171 self.send_levels()
|
|
172
|
|
173 subtk.slider_var.trace('w', slider_changed)
|
0
|
174 def make_row(self):
|
146
|
175 row = Frame(self, bd=2, bg='black')
|
0
|
176 row.pack(expand=1, fill=BOTH)
|
|
177 self.setup_key_nudgers(row)
|
|
178 self.rows.append(row)
|
|
179 return row
|
|
180 def draw_sub_slider(self, row, col, name, current_level):
|
|
181 subtk = SubmasterTk(row, name, current_level)
|
|
182 subtk.place(relx=col * 0.1, rely=0, relwidth=0.1, relheight=1)
|
|
183 self.setup_key_nudgers(subtk.scale)
|
|
184
|
|
185 self.slider_vars[name] = subtk.slider_var
|
|
186 return subtk
|
|
187 def highlight_row(self, row):
|
|
188 row = self.rows[row]
|
|
189 row['bg'] = 'red'
|
|
190 def unhighlight_row(self, row):
|
|
191 row = self.rows[row]
|
146
|
192 row['bg'] = 'black'
|
0
|
193 def get_levels(self):
|
|
194 return dict([(name, slidervar.get())
|
|
195 for name, slidervar in self.slider_vars.items()])
|
146
|
196 def get_levels_as_sub(self):
|
136
|
197 scaledsubs = [self.submasters.get_sub_by_name(sub) * level \
|
|
198 for sub, level in self.get_levels().items()]
|
|
199
|
|
200 maxes = sub_maxes(*scaledsubs)
|
146
|
201 return maxes
|
|
202 def save_current_stage(self, subname):
|
|
203 print "saving current levels as", subname
|
|
204 sub = self.get_levels_as_sub()
|
|
205 sub.name = subname
|
|
206 sub.save()
|
|
207
|
136
|
208 def save(self):
|
|
209 pickle.dump(self.get_levels(),
|
|
210 file('.keyboardcomposer.savedlevels', 'w'))
|
|
211 def send_frequent_updates(self):
|
|
212 """called when we get a fade -- send events as quickly as possible"""
|
|
213 if time.time() <= self.stop_frequent_update_time:
|
|
214 self.send_levels()
|
|
215 self.after(10, self.send_frequent_updates)
|
146
|
216
|
|
217 def get_dmx_list(self):
|
|
218 maxes = self.get_levels_as_sub()
|
|
219 return maxes.get_dmx_list()
|
136
|
220 def send_levels(self):
|
148
|
221 if not self.dmxdummy:
|
|
222 levels = self.get_dmx_list()
|
|
223 dmxclient.outputlevels(levels)
|
138
|
224 # print "sending levels", levels
|
136
|
225 def send_levels_loop(self):
|
|
226 self.send_levels()
|
|
227 self.after(1000, self.send_levels_loop)
|
|
228 def refresh(self):
|
|
229 self.save()
|
|
230 self.submasters = Submasters()
|
|
231 self.current_sub_levels = \
|
|
232 pickle.load(file('.keyboardcomposer.savedlevels'))
|
|
233 for r in self.rows:
|
|
234 r.destroy()
|
|
235 self.keyhints.destroy()
|
146
|
236 self.buttonframe.destroy()
|
136
|
237 self.draw_ui()
|
0
|
238
|
|
239 if __name__ == "__main__":
|
136
|
240 s = Submasters()
|
0
|
241
|
|
242 root = Tk()
|
136
|
243 tl = toplevelat("Keyboard Composer", existingtoplevel=root)
|
148
|
244 kc = KeyboardComposer(tl, s, dmxdummy=1)
|
0
|
245 kc.pack(fill=BOTH, expand=1)
|
136
|
246 atexit.register(kc.save)
|
|
247 try:
|
|
248 mainloop()
|
|
249 except KeyboardInterrupt:
|
|
250 tl.destroy()
|
|
251 sys.exit()
|