changeset 161:0803fb42109d

we now have TkCueList, which is really cool. it doesn't provide editing we now have TkCueList, which is really cool. it doesn't provide editing yet, but you could almost nearly probably maybe run a show with it. heck, i hope so. some of the shifting/drawing problems were probably fixed. cuelist1 got more bogus data to help populate the TkCueList.
author dmcc
date Mon, 07 Jul 2003 17:18:26 +0000
parents c1c0f5854f8f
children 6c5753bad783
files flax/CueFaders.py flax/cues/cuelist1
diffstat 2 files changed, 209 insertions(+), 65 deletions(-) [+]
line wrap: on
line diff
--- a/flax/CueFaders.py	Mon Jul 07 08:45:39 2003 +0000
+++ b/flax/CueFaders.py	Mon Jul 07 17:18:26 2003 +0000
@@ -2,6 +2,14 @@
 import Tix as Tk
 import time
 from TreeDict import TreeDict, allow_class_to_be_pickled
+from TLUtility import enumerate
+
+cue_state_indicator_colors = {
+             # bg       fg
+    'prev' : ('blue',   'white'),
+    'cur' :  ('yellow', 'black'),
+    'next' : ('red',    'white'),
+}
 
 class LabelledScale(Tk.Frame):
     """Scale with two labels: a name and current value"""
@@ -9,6 +17,11 @@
         Tk.Frame.__init__(self, master, bd=2, relief='raised')
         opts.setdefault('variable', Tk.DoubleVar())
         opts.setdefault('showvalue', 0)
+
+        self.normaltrough = opts.get('troughcolor', 'black')
+        self.flashtrough = opts.get('flashtroughcolor', 'red')
+        del opts['flashtroughcolor']
+
         self.scale_var = opts['variable']
         self.scale = Tk.Scale(self, **opts)
         self.scale.pack(side='top', expand=1, fill='both')
@@ -24,6 +37,10 @@
     def update_value_label(self, *args):
         val = self.scale_var.get() * 100
         self.scale_value['text'] = "%0.2f" % val
+        if val != 0:
+            self.scale['troughcolor'] = self.flashtrough
+        else:
+            self.scale['troughcolor'] = self.normaltrough
     def disable(self):
         if not self.disabled:
             self.scale['state'] = 'disabled'
@@ -36,11 +53,11 @@
 
 class TimedGoButton(Tk.Frame):
     """Go button, fade time entry, and time fader"""
-    def __init__(self, master, name, scale_to_fade):
+    def __init__(self, master, name, scale_to_fade, **kw):
         Tk.Frame.__init__(self, master)
         self.name = name
         self.scale_to_fade = scale_to_fade
-        self.button = Tk.Button(self, text=name, command=self.start_fade)
+        self.button = Tk.Button(self, text=name, command=self.start_fade, **kw)
         self.button.pack(fill='both', expand=1, side='left')
         self.timer_var = Tk.StringVar()
         self.timer_entry = Tk.Entry(self, textvariable=self.timer_var, width=5)
@@ -88,36 +105,51 @@
         self.auto_shift = Tk.IntVar()
         self.auto_shift.set(1)
 
+        # this is a mechanism to stop Tk from autoshifting too much.
+        # if this variable is true, the mouse button is down.  we don't want
+        # to shift until they release it.  when it is released, we will
+        # set it to false and then call autoshift.
+        self.no_shifts_until_release = 0
+
         self.scales = {}
         self.shift_buttons = {}
         self.go_buttons = {}
-
+        
         topframe = Tk.Frame(self)
-        self.current_cues = Tk.Label(topframe)
-        self.current_cues.pack()
-        self.update_cue_display()
-        topframe.pack()
-        
-        bottomframe = Tk.Frame(self)
-        self.auto_shift_checkbutton = Tk.Checkbutton(self, 
+
+        self.set_prev_button = Tk.Button(topframe, text='Set Prev',
+            command=lambda: cuelist.set_selection_as_prev(),
+            fg='white', bg='blue')
+        self.set_prev_button.pack(side='left')
+
+        self.auto_shift_checkbutton = Tk.Checkbutton(topframe, 
             variable=self.auto_shift, text='Autoshift', 
             command=self.toggle_autoshift)
-        self.auto_shift_checkbutton.pack()
-        bottomframe.pack(side='bottom')
+        self.auto_shift_checkbutton.pack(side='left')
+
+        self.set_next_button = Tk.Button(topframe, text='Set Next',
+            command=lambda: cuelist.set_selection_as_next(),
+            fg='white', bg='red')
+        self.set_next_button.pack(side='left')
 
-        middleframe = Tk.Frame(self)
-        for name, start, end, side in (('Prev', 1, 0, 'left'),
-                                       ('Next', 0, 1, 'right')):
+        topframe.pack(side='top')
+
+        faderframe = Tk.Frame(self)
+        self.direction_info = (('Prev', 1, 0, 'left', 'blue'),
+                               ('Next', 0, 1, 'right', 'red'))
+        for name, start, end, side, color in self.direction_info:
             frame = Tk.Frame(self)
             scale = LabelledScale(frame, name, from_=start, to_=end, 
-                res=0.01, orient='horiz')
-            scale.pack(fill='both', expand=1)
-            go = TimedGoButton(frame, 'Go %s' % name, scale)
+                res=0.0001, orient='horiz', flashtroughcolor=color)
+            scale.pack(fill='x', expand=0)
+            go = TimedGoButton(frame, 'Go %s' % name, scale, bg=color, 
+                fg='white')
             go.pack(fill='both', expand=1)
             frame.pack(side=side, fill='both', expand=1)
         
             shift = Tk.Button(frame, text="Shift %s" % name, state='disabled',
-                command=lambda name=name: self.shift(name))
+                command=lambda name=name: self.shift(name), fg=color, 
+                bg='black')
 
             self.scales[name] = scale
             self.shift_buttons[name] = shift
@@ -125,37 +157,47 @@
 
             scale.scale_var.trace('w', \
                 lambda x, y, z, name=name, scale=scale: self.xfade(name, scale))
-        middleframe.pack(side='bottom', fill='both', expand=1)
+
+            def button_press(event, name=name, scale=scale):
+                self.no_shifts_until_release = 1 # prevent shifts until release
+            def button_release(event, name=name, scale=scale):
+                self.no_shifts_until_release = 0
+                self.autoshift(name, scale)
+
+            scale.scale.bind("<ButtonPress>", button_press)
+            scale.scale.bind("<ButtonRelease>", button_release)
+        faderframe.pack(side='bottom', fill='both', expand=1)
     def toggle_autoshift(self):
         for name, button in self.shift_buttons.items():
             if not self.auto_shift.get():
                 button.pack(side='bottom', fill='both', expand=1)
             else:
                 button.pack_forget()
+    def shift(self, name):
+        # to prevent overshifting
+        if self.no_shifts_until_release:
+            return
 
-    def shift(self, name):
-        print "shift", name
         for scale_name, scale in self.scales.items():
-            scale.scale_var.set(0)
+            scale.scale.set(0)
         self.cuelist.shift((-1, 1)[name == 'Next'])
-        self.update_cue_display()
-    def update_cue_display(self):
-        current_cues = [cue.name for cue in self.cuelist.get_current_cues()]
-        self.current_cues['text'] = ', '.join(current_cues)
-    def xfade(self, name, scale):
+    def autoshift(self, name, scale):
         scale_val = scale.scale_var.get() 
-        # print "xfade", name, scale_val
 
         if scale_val == 1:
             if self.auto_shift.get():
-                print "autoshifting", name
                 self.shift(name)
-                scale_val = scale.scale_var.get() # this needs to be refreshed
+    def xfade(self, name, scale):
+        if self.auto_shift.get():
+            self.autoshift(name, scale)
+            scale_val = scale.scale_var.get() 
+        else:
+            scale_val = scale.scale_var.get() 
+            if scale_val == 1:
+                self.shift_buttons[name]['state'] = 'normal'
             else:
-                self.shift_buttons[name]['state'] = 'normal'
-        else:
-            # disable any dangerous shifting
-            self.shift_buttons[name]['state'] = 'disabled'
+                # disable any dangerous shifting
+                self.shift_buttons[name]['state'] = 'disabled'
 
         d = self.opposite_direction(name)
         if scale_val != 0:
@@ -249,16 +291,112 @@
         self.save()
     def save(self):
         self.treedict.save(self.filename)
+    def reload(self):
+        # TODO: we probably will need to make sure that indices still make
+        # sense, etc.
+        self.treedict.load(self.filename)
+
+class TkCueList(CueList, Tk.Frame):
+    def __init__(self, master, filename):
+        CueList.__init__(self, filename)
+        Tk.Frame.__init__(self, master)
+        
+        self.columns = ('name', 'time', 'page', 'desc')
+            
+        self.scrolled_hlist = Tk.ScrolledHList(self,
+            options='hlist.columns %d hlist.header 1' % len(self.columns))
+        self.hlist = self.scrolled_hlist.hlist
+        self.hlist.configure(fg='white', bg='black', 
+            command=self.select_callback)
+        self.scrolled_hlist.pack(fill='both', expand=1)
+
+        boldfont = self.tk.call('tix', 'option', 'get', 
+            'bold_font')
+        header_style = Tk.DisplayStyle('text', refwindow=self,
+            anchor='center', padx=8, pady=2, font=boldfont)
+
+        for count, header in enumerate(self.columns):
+            self.hlist.header_create(count, itemtype='text',
+                text=header, style=header_style)
+
+        self.cue_label_windows = {}
+        for count, cue in enumerate(self.cues):
+            self.display_cue(count, cue)
+        self.update_cue_indicators()
+    def display_cue(self, cue_count, cue):
+        # cue_count is the path in the hlist -- this probably isn't ideal
+        for col, header in enumerate(self.columns):
+            try:
+                text = getattr(cue, header)
+            except AttributeError:
+                text = ''
+
+            if col == 0:
+                lab = Tk.Label(self.hlist, text=text, fg='white', bg='black')
+                def select_and_highlight(event):
+                    self.select_callback(cue_count)
+                    self.hlist.selection_clear()
+                    self.hlist.selection_set(cue_count)
+
+                lab.bind("<Double-1>", select_and_highlight)
+                self.hlist.add(cue_count, itemtype='window', window=lab)
+                self.cue_label_windows[cue_count] = lab
+            else:
+                self.hlist.item_create(cue_count, col, text=text)
+    def reset_cue_indicators(self, cue_indices=None):
+        """If cue_indices is None, we'll reset all of them."""
+        cue_indices = cue_indices or self.cue_label_windows.keys()
+        for key in cue_indices:
+            window = self.cue_label_windows[key]
+            window.configure(fg='white', bg='black')
+    def update_cue_indicators(self):
+        states = dict(zip(self.get_current_cue_indices(), 
+                     ('prev', 'cur', 'next')))
+
+        for count, state in states.items():
+            window = self.cue_label_windows[count]
+            bg, fg = cue_state_indicator_colors[state]
+            window.configure(bg=bg, fg=fg)
+    def shift(self, diff):
+        self.reset_cue_indicators(self.get_current_cue_indices())
+        CueList.shift(self, diff)
+        self.update_cue_indicators()
+        # try to see all indices, but next takes priority over all, and cur
+        # over prev
+        for index in self.get_current_cue_indices():
+            self.hlist.see(index)
+    def select_callback(self, index):
+        new_next = int(index)
+        self.set_next(new_next)
+    def set_next(self, index):
+        prev, cur, next = self.get_current_cue_indices()
+        self.reset_cue_indicators((next,))
+        CueList.set_next(self, index)
+        self.update_cue_indicators()
+    def set_prev(self, index):
+        prev, cur, next = self.get_current_cue_indices()
+        self.reset_cue_indicators((prev,))
+        CueList.set_prev(self, index)
+        self.update_cue_indicators()
+    def set_selection_as_prev(self):
+        sel = self.hlist.info_selection()
+        if sel:
+            self.set_prev(int(sel[0]))
+    def set_selection_as_next(self):
+        sel = self.hlist.info_selection()
+        if sel:
+            self.set_next(int(sel[0]))
 
 if __name__ == "__main__":
-    cl = CueList('cues/cuelist1')
+    root = Tk.Tk()
+    cl = TkCueList(root, 'cues/cuelist1')
+    cl.pack(fill='both', expand=1)
 
     # to populate cue list
     if 0:
         for x in range(20):
             cl.add_cue(Cue('cue %d' % x, time=x, some_attribute=3))
 
-    root = Tk.Tk()
     fader = CueFader(root, cl)
     fader.pack(fill='both', expand=1)
     Tk.mainloop()
--- a/flax/cues/cuelist1	Mon Jul 07 08:45:39 2003 +0000
+++ b/flax/cues/cuelist1	Mon Jul 07 17:18:26 2003 +0000
@@ -1,106 +1,112 @@
 <?xml version="1.0"?>
 <!DOCTYPE PyObject SYSTEM "PyObjects.dtd">
 <PyObject family="obj" type="builtin_wrapper"  class="_EmptyClass">
-<attr name="__toplevel__" family="map" type="__compound__" extra="None TreeDict" id="135916956" >
+<attr name="__toplevel__" family="map" type="__compound__" extra="None TreeDict" id="138321692" >
   <entry>
     <key type="string" value="cues" />
-    <val type="list" id="138285508" >
-      <item type="PyObject" id="138620316" class="Cue">
+    <val type="list" id="140057044" >
+      <item type="PyObject" id="139705788" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 0" />
         <attr name="time" type="numeric" value="0" />
       </item>
-      <item type="PyObject" id="138232236" class="Cue">
+      <item type="PyObject" id="138399700" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 1" />
         <attr name="time" type="numeric" value="1" />
       </item>
-      <item type="PyObject" id="138598276" class="Cue">
+      <item type="PyObject" id="139651028" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 2" />
         <attr name="time" type="numeric" value="2" />
       </item>
-      <item type="PyObject" id="138229380" class="Cue">
+      <item type="PyObject" id="139651716" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 3" />
         <attr name="time" type="numeric" value="3" />
       </item>
-      <item type="PyObject" id="138309300" class="Cue">
+      <item type="PyObject" id="140056788" class="Cue">
+        <attr name="time" type="numeric" value="4" />
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 4" />
-        <attr name="time" type="numeric" value="4" />
+        <attr name="page" type="string" value="1.3.13" />
       </item>
-      <item type="PyObject" id="138229564" class="Cue">
+      <item type="PyObject" id="139731244" class="Cue">
+        <attr name="time" type="numeric" value="5" />
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 5" />
-        <attr name="time" type="numeric" value="5" />
+        <attr name="page" type="string" value="1.3.17" />
       </item>
-      <item type="PyObject" id="138296364" class="Cue">
+      <item type="PyObject" id="139724292" class="Cue">
+        <attr name="time" type="numeric" value="6" />
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 6" />
-        <attr name="time" type="numeric" value="6" />
+        <attr name="page" type="string" value="1.4.2" />
       </item>
-      <item type="PyObject" id="138256796" class="Cue">
+      <item type="PyObject" id="138615364" class="Cue">
+        <attr name="time" type="numeric" value="7" />
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 7" />
-        <attr name="time" type="numeric" value="7" />
+        <attr name="page" type="string" value="2.1.1" />
       </item>
-      <item type="PyObject" id="136370908" class="Cue">
+      <item type="PyObject" id="139653716" class="Cue">
+        <attr name="time" type="numeric" value="8" />
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 8" />
-        <attr name="time" type="numeric" value="8" />
+        <attr name="page" type="string" value="2.1.2" />
       </item>
-      <item type="PyObject" id="136982724" class="Cue">
+      <item type="PyObject" id="140059964" class="Cue">
+        <attr name="time" type="numeric" value="9" />
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 9" />
-        <attr name="time" type="numeric" value="9" />
+        <attr name="page" type="string" value="2.2.4" />
       </item>
-      <item type="PyObject" id="138256916" class="Cue">
+      <item type="PyObject" id="139654716" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 10" />
         <attr name="time" type="numeric" value="10" />
       </item>
-      <item type="PyObject" id="138233236" class="Cue">
+      <item type="PyObject" id="139654900" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 11" />
         <attr name="time" type="numeric" value="11" />
       </item>
-      <item type="PyObject" id="138233420" class="Cue">
+      <item type="PyObject" id="139655084" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 12" />
         <attr name="time" type="numeric" value="12" />
       </item>
-      <item type="PyObject" id="138233660" class="Cue">
+      <item type="PyObject" id="139655124" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 13" />
         <attr name="time" type="numeric" value="13" />
       </item>
-      <item type="PyObject" id="138296580" class="Cue">
+      <item type="PyObject" id="139655364" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 14" />
         <attr name="time" type="numeric" value="14" />
       </item>
-      <item type="PyObject" id="136859228" class="Cue">
+      <item type="PyObject" id="139655548" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 15" />
         <attr name="time" type="numeric" value="15" />
       </item>
-      <item type="PyObject" id="138233700" class="Cue">
+      <item type="PyObject" id="139655588" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 16" />
         <attr name="time" type="numeric" value="16" />
       </item>
-      <item type="PyObject" id="138233740" class="Cue">
+      <item type="PyObject" id="140051868" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 17" />
         <attr name="time" type="numeric" value="17" />
       </item>
-      <item type="PyObject" id="138609044" class="Cue">
+      <item type="PyObject" id="139652364" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 18" />
         <attr name="time" type="numeric" value="18" />
       </item>
-      <item type="PyObject" id="138230836" class="Cue">
+      <item type="PyObject" id="140041636" class="Cue">
         <attr name="some_attribute" type="numeric" value="3" />
         <attr name="name" type="string" value="cue 19" />
         <attr name="time" type="numeric" value="19" />