Mercurial > code > home > repos > light9
annotate light8/ExtSliderMapper.py @ 101:a995fd1a8f03
result of 7.12 performance
author | dmcc |
---|---|
date | Sun, 14 Jul 2002 02:44:27 +0000 |
parents | f13cf18224f2 |
children | e04f7b552bcd |
rev | line source |
---|---|
100 | 1 """ The External Slider Mapping widget determines which pots map to which |
2 submasters. It tells you the status of each mapping and saves and loads | |
3 presets. The show is relying on this module! Do not lose it! | |
4 | |
101 | 5 FUQ (frequently unasked question(s)) |
100 | 6 |
7 1. What's with all the *args? | |
8 | |
9 It lets functions take any number of arguments and throw them away. | |
10 Callbacks do this, and we typically don't care about what they have to say. """ | |
11 | |
0 | 12 from Tix import * |
101 | 13 from uihelpers import FancyDoubleVar |
14 | |
15 stdfont = ('Arial', 8) | |
0 | 16 |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
17 class SliderMapping: |
92 | 18 def __init__(self, default='disconnected', synced=0, extinputlevel=0, |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
19 sublevel=0): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
20 self.subname = StringVar() # name of submaster we're connected to |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
21 self.subname.set(default) |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
22 self.sublevel = DoubleVar() # scalelevel variable of that submaster |
101 | 23 # self.sublevel = FancyDoubleVar() # scalelevel variable of that submaster |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
24 self.sublevel.set(sublevel) |
92 | 25 self.synced = BooleanVar() # currently synced |
26 self.synced.set(synced) | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
27 self.extlevel = DoubleVar() # external slider's input |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
28 self.extlevel.set(extinputlevel) |
87 | 29 self.widgets = [] # list of widgets drawn |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
30 self.sublabel = None # the label which represents a sub level. |
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
31 # we hold on to it so we can change its |
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
32 # textvariable |
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
33 self.statuslabel = None # tells us sync status |
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
34 self.lastbgcolor = None # last background color drawn to avoid |
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
35 # unnecessary redraws |
92 | 36 def sync(self, *args): |
37 self.synced.set(1) | |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
38 self.color_bg() |
92 | 39 def unsync(self, *args): |
40 self.synced.set(0) | |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
41 self.color_bg() |
92 | 42 def issynced(self): |
43 return self.synced.get() | |
44 def disconnect(self, *args): | |
45 self.set_subname('disconnected') # a bit hack-like | |
101 | 46 # self.sublevel.delete_named('sync') |
47 ''' | |
48 try: | |
49 if self.sublevel.unsync_trace_cbname is not None: | |
50 # self.sublevel.trace_vdelete('w', | |
51 # self.sublevel.unsync_trace_cbname) | |
52 self.sublevel._tk.call('trace', 'vdelete', self.sublevel._name, | |
53 'w', self.sublevel.unsync_trace_cbname) | |
54 self.sublevel.unsync_trace_cbname = None | |
55 except AttributeError: | |
56 pass | |
57 ''' | |
58 | |
92 | 59 self.sublabel.configure(text="N/A") |
60 self.color_bg() | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
61 def isdisconnected(self): |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
62 return self.subname.get() == 'disconnected' # a bit more hack-like |
92 | 63 def check_synced(self, *args): |
64 'If external level is near than the sublevel, it synces' | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
65 if self.isdisconnected(): |
92 | 66 self.unsync() |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
67 return |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
68 |
92 | 69 if abs(self.extlevel.get() - self.sublevel.get()) <= 0.02: |
70 self.sync() | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
71 def changed_extinput(self, newlevel): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
72 'When a new external level is received, this incorporates it' |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
73 self.extlevel.set(newlevel) |
92 | 74 self.check_synced() |
75 self.color_bg() | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
76 def set_subname(self, newname): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
77 self.subname.set(newname) |
92 | 78 self.unsync() |
79 self.color_bg() | |
80 def color_bg(self): | |
87 | 81 if self.widgets: |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
82 if self.isdisconnected(): |
87 | 83 color = 'honeyDew4' |
96 | 84 # stupid hack |
99 | 85 # elif abs(self.extlevel.get() - self.sublevel.get()) <= 0.02: |
100 | 86 elif self.issynced(): |
87 | 87 color = 'honeyDew2' |
92 | 88 else: # unsynced |
87 | 89 color = 'red' |
90 | |
100 | 91 if self.statuslabel: # more stupid hackery |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
92 if color == 'honeyDew2': # connected |
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
93 self.statuslabel['text'] = 'Sync' |
99 | 94 elif self.extlevel.get() < self.sublevel.get(): |
95 self.statuslabel['text'] = 'No sync (go up)' | |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
96 else: |
99 | 97 self.statuslabel['text'] = 'No sync (go down)' |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
98 |
92 | 99 # print "color", color, "lastbgcolor", self.lastbgcolor |
87 | 100 if self.lastbgcolor == color: return |
101 for widget in self.widgets: | |
102 widget.configure(bg=color) | |
103 self.lastbgcolor = color | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
104 def set_sublevel_var(self, newvar): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
105 'newvar is one of the variables in scalelevels' |
98 | 106 |
107 if newvar is not self.sublevel: | |
101 | 108 # self.sublevel.delete_named('sync') |
109 self.sublevel = newvar | |
110 self.sublabel.configure(textvariable=newvar) | |
111 # self.sublevel.trace_named('sync', lambda *args: self.unsync(*args)) | |
112 ''' | |
98 | 113 try: |
101 | 114 if self.sublevel.unsync_trace_cbname is not None: |
115 # remove an old trace | |
116 self.sublevel.trace_vdelete('w', | |
117 self.sublevel.unsync_trace_cbname) | |
98 | 118 except AttributeError: |
119 pass # it didn't have one | |
120 | |
121 self.sublevel = newvar | |
100 | 122 self.sublevel.unsync_trace_cbname = self.sublevel.trace('w', |
123 self.unsync) | |
101 | 124 ''' |
98 | 125 |
101 | 126 # if self.sublabel: |
127 # self.sublabel.configure(textvariable=newvar) | |
92 | 128 self.check_synced() |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
129 def get_mapping(self): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
130 'Get name of submaster currently mapped' |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
131 return self.subname.get() |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
132 def get_level_pair(self): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
133 'Returns suitable output for ExtSliderMapper.get_levels()' |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
134 return (self.subname.get(), self.extlevel.get()) |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
135 def draw_interface(self, master, subnames): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
136 'Draw interface into master, given a list of submaster names' |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
137 frame = Frame(master) |
101 | 138 c = ComboBox(frame, variable=self.subname, dropdown=0) |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
139 c.slistbox.listbox.insert(END, "disconnected") |
101 | 140 # c.listbox.insert(END, "disconnected") |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
141 for s in subnames: |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
142 c.slistbox.listbox.insert(END, s) |
101 | 143 # c.listbox.insert(END, s) |
144 c.entry.configure(width=12, font=stdfont) | |
145 c.slistbox.listbox.configure(font=stdfont, exportselection=0) | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
146 statframe = Frame(frame) |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
147 |
101 | 148 self.statuslabel = Label(statframe, text="No sync", width=15, font=stdfont) |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
149 self.statuslabel.grid(columnspan=2, sticky=W) |
101 | 150 ilabel = Label(statframe, text="Input", fg='blue', font=stdfont) |
87 | 151 ilabel.grid(row=1, sticky=W) |
101 | 152 extlabel = Label(statframe, textvariable=self.extlevel, width=5, font=stdfont) |
87 | 153 extlabel.grid(row=1, column=1) |
101 | 154 rlabel = Label(statframe, text="Real", font=stdfont) |
87 | 155 rlabel.grid(row=2, sticky=W) |
101 | 156 self.sublabel = Label(statframe, text="N/A", width=5, font=stdfont) |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
157 self.sublabel.grid(row=2, column=1) |
101 | 158 disc_button = Button(statframe, text="Disconnect", |
159 command=self.disconnect, padx=0, pady=0, font=stdfont) | |
160 disc_button.grid(row=3, columnspan=2) | |
161 statframe.pack(side=BOTTOM, expand=1, fill=BOTH) | |
162 c.pack(expand=1, fill=BOTH) | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
163 frame.pack(side=LEFT, expand=1, fill=BOTH) |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
164 |
94
29a8b23d8db5
ready for better scalelevel variable, checkbutton -> label
dmcc
parents:
92
diff
changeset
|
165 self.widgets = [frame, c, statframe, self.statuslabel, ilabel, extlabel, |
101 | 166 rlabel, self.sublabel, disc_button] |
87 | 167 |
0 | 168 class ExtSliderMapper(Frame): |
80 | 169 def __init__(self, parent, sliderlevels, sliderinput, filename='slidermapping', |
170 numsliders=4): | |
0 | 171 'Slider levels is scalelevels, sliderinput is an ExternalInput object' |
80 | 172 Frame.__init__(self, parent) |
0 | 173 self.parent = parent |
174 self.sliderlevels = sliderlevels | |
175 self.sliderinput = sliderinput | |
176 self.filename = filename | |
177 self.numsliders = numsliders | |
178 self.file = None | |
179 | |
180 # don't call setup, let them do that when scalelevels is created | |
181 def setup(self): | |
182 self.subnames = self.sliderlevels.keys() | |
183 self.subnames.sort() | |
184 self.presets = {} | |
185 self.load_presets() | |
186 | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
187 self.current_preset = StringVar() # name of current preset |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
188 self.current_mappings = [] |
80 | 189 for i in range(self.numsliders): |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
190 self.current_mappings.append(SliderMapping()) |
0 | 191 |
192 self.draw_interface() | |
80 | 193 def load_presets(self): |
101 | 194 self.presets = {} |
0 | 195 self.file = open(self.filename, 'r') |
196 lines = self.file.readlines() | |
197 for l in lines: | |
198 tokens = l[:-1].split('\t') | |
199 name = tokens.pop(0) | |
200 self.presets[name] = tokens | |
201 self.file.close() | |
202 def save_presets(self): | |
203 self.file = open(self.filename, 'w') | |
204 self.file.seek(0) | |
205 preset_names = self.presets.keys() | |
206 preset_names.sort() | |
207 for p in preset_names: | |
208 s = '\t'.join([p] + self.presets[p]) + '\n' | |
209 self.file.write(s) | |
210 self.file.close() | |
211 def load_scalelevels(self): | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
212 for slidermap in self.current_mappings: |
0 | 213 try: |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
214 v = self.sliderlevels[slidermap.get_mapping()] |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
215 slidermap.set_sublevel_var(v) |
0 | 216 except KeyError: |
101 | 217 name = slidermap.get_mapping() |
218 if name != 'disconnected': | |
219 print "ESM: No submaster named", name | |
0 | 220 |
221 def get_levels(self): | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
222 'Called by changelevels, returns a dict of new values for submasters' |
0 | 223 if not self.sliderinput: return {} |
224 | |
101 | 225 self.load_scalelevels() # freshen our input from the submasters |
0 | 226 |
227 rawlevels = self.sliderinput.get_levels() | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
228 for rawlev, slidermap in zip(rawlevels, self.current_mappings): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
229 slidermap.changed_extinput(rawlev) |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
230 |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
231 return dict([m.get_level_pair() |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
232 for m in self.current_mappings |
92 | 233 if m.issynced()]) |
0 | 234 def draw_interface(self): |
80 | 235 self.reallevellabels = [] |
236 subchoiceframe = Frame(self) | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
237 for m in self.current_mappings: |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
238 m.draw_interface(subchoiceframe, self.subnames) |
0 | 239 subchoiceframe.pack() |
240 | |
80 | 241 presetframe = Frame(self) |
242 Label(presetframe, text="Preset:").pack(side=LEFT) | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
243 self.presetcombo = ComboBox(presetframe, variable=self.current_preset, |
80 | 244 editable=1, command=self.apply_preset) |
0 | 245 self.draw_presets() |
246 self.presetcombo.pack(side=LEFT) | |
80 | 247 Button(presetframe, text="Add", padx=0, pady=0, |
0 | 248 command=self.add_preset).pack(side=LEFT) |
80 | 249 Button(presetframe, text="Delete", padx=0, pady=0, |
0 | 250 command=self.delete_preset).pack(side=LEFT) |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
251 Button(presetframe, text="Disconnect", padx=0, pady=0, |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
252 command=self.disconnect_all).pack(side=LEFT) |
101 | 253 Button(presetframe, text="Reload", padx=0, pady=0, |
254 command=self.load_presets).pack(side=LEFT) | |
0 | 255 presetframe.pack(side=BOTTOM) |
256 def apply_preset(self, preset): | |
257 if not preset: return | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
258 preset_mapping = self.presets.get(preset) |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
259 if not preset_mapping: return |
101 | 260 self.disconnect_all() |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
261 for subname, slidermap in zip(preset_mapping, self.current_mappings): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
262 slidermap.set_subname(subname) |
0 | 263 def delete_preset(self, *args): |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
264 del self.presets[self.current_preset.get()] |
0 | 265 self.presetcombo.slistbox.listbox.delete(0, END) |
266 self.draw_presets() | |
267 self.save_presets() | |
268 def add_preset(self, *args): | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
269 self.presets[self.current_preset.get()] = [m.get_mapping() |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
270 for m in self.current_mappings] |
0 | 271 self.presetcombo.slistbox.listbox.delete(0, END) |
272 self.draw_presets() | |
273 self.save_presets() | |
274 def draw_presets(self): | |
275 preset_names = self.presets.keys() | |
276 preset_names.sort() | |
277 for p in preset_names: | |
278 self.presetcombo.slistbox.listbox.insert(END, p) | |
86
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
279 def disconnect_all(self): |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
280 for m in self.current_mappings: |
5a162150b68d
new slidermappings, extslidermapping modularized and slightly smarter
dmcc
parents:
80
diff
changeset
|
281 m.disconnect() |