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