How to create a sliding widget (GTK+, Quickly)

Solution 1:

I have achieved a fade-out effect using pure GTK and CSS.

It is only working in GTK 3.6 and I am not sure if a slide in / out effect would be possible, however, if you want, you can look at the source at launchpad.net/uberwriter

It works through a state-change and then GTK Transitions... Maybe with height // width that would be possible, too.

EDIT

Because people obviously downvote me, here is another, more detailed explanation:

This is the CSS I have used:

GtkLabel:insensitive {
    color: rgba(255,255,255,0);
    transition: 500ms ease-in-out;
}

If you use this as CSS (I hope I am allowed to reference an explanation from myself, how to that: http://wolfvollprecht.de/blog/gtk-python-and-css-are-an-awesome-combo/) then you can use Gtk.StateFlags to fade out the Label.

e.g.:

label.set_state_flags(Gtk.StateFlags.INSENSITIVE, True)

Then you get a transition to zero opacity.

However, as I have noticed, there are not many options to influence placement / width with CSS, so I guess it's limited to colors / opacity etc.

Enjoy.

PS: Note that this only works in Gtk 3.6, since Ubuntu 12.10

Solution 2:

Such animation is not achievable in pure GTK+. There are no mechanisms that would process it.

I have given a quick look on gedit's source code, it's clear that they process this animation on their own. I have not looked into details, for the code related to animations is quite expanded, but I have noted that they incorporate Cairo drawing features for the animated search widget. Most likely the widget is animated by moving its position frame by frame and redrawing it on a different place in the overlay that is used on a single text display area.

That seems to be the simplest solution. (gedit's code seems to be prepared for many animations sharing the same code base, thus it may be an overkill to use it's exact approach in a simple application.)

To reproduce this effect, you will need to:

  • Use a custom widget for the UI piece that you want to slide in/out.
  • Make it react on draw event and periodical timeouts (to redraw each frame of animation)
  • Create a transparent overlay over the rest of your widgets (or any area that has to appear as if it was below the sliding widget)
  • Draw the animated widget manually on such overlay.

I can't come up with any easier solution. However, considering the fact that gedit developers do know GTK+ as probably very little do, there may be no simpler trick to achieve such effect.

Solution 3:

You can use a combination between Gtk.fixed(),GObject.timeout_add and the function move, here is an example in python :

#!/usr/bin/python*
from gi.repository import Gtk, GObject

class TestWindow(Gtk.Window):
     def animateImage(self):
        GObject.timeout_add(150,self.slideImage)

     def slideImage(self):
        self.positionX += 50;
        if(self.positionX > 800):
          self.fixedWidget.move(self.logo,self.positionX,200)
          return True        
        else:
          return False 

     def __init__(self):
        Gtk.Window.__init__(self, title='Registration')

        self.positionX = 500
        self.fixedWidget = Gtk.Fixed()
        self.fixedWidget.set_size_request(1920,1080)
        self.fixedWidget.show()

        self.logo = Gtk.Image.new_from_file('picture.png')
        self.logo.show()
        self.fixedWidget.put(self.logo,self.positionX,200)

        self.button1 = Gtk.Button('Click me to slide image!')
        self.button1.show()
        self.button1.connect('clicked', self.animateImage)
        self.button1.set_size_request(75,30)
        self.fixedWidget.put(self.button1,750,750)
        self.add(self.fixedWidget)

testWindow = TestWindow()
testWindow.set_size_request(1920,1080)
testWindow.set_name('testWindow')
testWindow.show()

Gtk.main()

Solution 4:

Nowadays, there is a real answer to this question. Since it was originally asked and answered, the code for revealing a widget from libgd - which I presume is what GEdit was using at that time - has been upstreamed into GTK+ as GtkRevealer:

https://developer.gnome.org/gtk3/stable/GtkRevealer.html

GtkRevealer supports various forms of transition, including the cardinal directions of slide and a fade on the opacity.

So, these days, it's as simple as adding the widget that you want to transition to a GtkRevealer and using its :transition-(type|duration) and :reveal-child properties.