12

I'm using the slider to update my visualization, but the command updateValue is sent everytime I move the slider thumb, even for intermediate values.

Instead I want to trigger it only when I release the mouse button and the interaction is complete.

self.slider = tk.Scale(self.leftFrame, from_=0, to=256, orient=tk.HORIZONTAL, command=updateValue)

How can I trigger the function only once, when the interaction is ended ?

aneuryzm
  • 63,052
  • 100
  • 273
  • 488

2 Answers2

55

This is quite an ancient question now, but in case anyone stumbles upon this particular problem just use the bind() function and the "ButtonRelease-1" event like so:

import Tkinter as tk

class App:
    def __init__(self):
        self.root = tk.Tk()

        self.slider = tk.Scale(self.root, from_=0, to=256, 
                               orient="horizontal")
        self.slider.bind("<ButtonRelease-1>", self.updateValue)
        self.slider.pack()
        self.root.mainloop()

    def updateValue(self, event):
        print self.slider.get()

app=App()

Hope this helps anyone!

dsoosh
  • 693
  • 5
  • 9
  • I know this is a reply to an encient question, but it was exactly the answer I was looking for. Thank you. – mhourdakis Jun 16 '15 at 10:01
  • 11
    Beware! If Scale.takefocus is True, then people will be able to move the slider using just keyboard arrow keys. **But nothing will ever happen** because you are only catching mouse click. To remedy, catch the key presses too. (You can bind multiple events for a single callback) – AneesAhmed777 Jun 03 '17 at 12:18
9

You can't.

What you can do instead is have your command delay any real work for a short period of time using 'after'. Each time your command is called, cancel any pending work and reschedule the work. Depending on what your actual requirements are, a half second delay might be sufficient.

Another choice is to not use the built-in command feature and instead use custom bindings. This can be a lot of work to get exactly right, but if you really need fine grained control you can do it. Don't forget that one can interact with the widget using the keyboard in addition to the mouse.

Here's a short example showing how to schedule the work to be done in half a second:

import Tkinter as tk

#create window & frames
class App:
    def __init__(self):
        self.root = tk.Tk()
        self._job = None
        self.slider = tk.Scale(self.root, from_=0, to=256, 
                               orient="horizontal", 
                               command=self.updateValue)
        self.slider.pack()
        self.root.mainloop()

    def updateValue(self, event):
        if self._job:
            self.root.after_cancel(self._job)
        self._job = self.root.after(500, self._do_something)

    def _do_something(self):
        self._job = None
        print "new value:", self.slider.get()

app=App()
Bryan Oakley
  • 370,779
  • 53
  • 539
  • 685
  • Ok, what's the best way to have a 0.5 delay ? – aneuryzm Oct 19 '10 at 13:27
  • Thanks, I get this error: AttributeError: 'str' object has no attribute '_job' – aneuryzm Oct 19 '10 at 14:22
  • @Patric: I can't debug your code for you, especially without seeing it. Think about what the error message is telling you, this is not a hard problem. You are trying to access the attribute "_job" from some object. That object has no such attribute. Thus, either you're using the wrong object, or your object is somehow broken. – Bryan Oakley Oct 19 '10 at 15:43