diff --git a/doc/sources/guide/widgets.rst b/doc/sources/guide/widgets.rst index a212bbfd4..251b0fd34 100644 --- a/doc/sources/guide/widgets.rst +++ b/doc/sources/guide/widgets.rst @@ -135,106 +135,58 @@ its children. There are different kinds of layouts, allowing for different automatic organization of their children. Layouts use |size_hint| and |pos_hint| properties to determine the |size| and |pos| of their |children|. - - Layout types +**BoxLayout**: +Arranges widgets in a side to side (either vertically or horizontally) manner, +to fill all the place. +size_hint of children can be used to change proportions allowed to each +children, or set fixed size for some of them - **BoxLayout**: - Arranges widgets in a side-by-side (if the orientation is 'horizontal') or - top-to-bottom (if the orientaion is 'vertical') manner using all the space - available. The `size_hint` of the children can be used to change the proportions - of each child or set them to a fixed size. - - `pos_hint` is not honored. +.. only:: html - .. only:: html + .. image:: ../images/boxlayout.gif + :align: left + .. image:: ../images/gridlayout.gif + :align: right + .. image:: ../images/stacklayout.gif + :align: left + .. image:: ../images/anchorlayout.gif + :align: right + .. image:: ../images/floatlayout.gif - .. image:: ../images/boxlayout.gif - :align: right - :width: 200 +.. only:: latex - .. only:: latex + .. image:: ../images/boxlayout.png + .. image:: ../images/gridlayout.png + .. image:: ../images/stacklayout.png + .. image:: ../images/anchorlayout.png + .. image:: ../images/floatlayout.png - .. image:: ../images/boxlayout.png - :align: right - :width: 200 - **GridLayout**: - Arranges widgets in a grid. You must specifiy at least one dimension for the - grid so that kivy can compute the size of the elements and how to arrange them. +**GridLayout**: +Arranges widgets in a grid. You must specifiy at least one dimension of the +grid so kivy can compute the size of the elements and how to arrange them. - `pos_hint` is not honored. +**StackLayout**: +Arranges widgets side to side, but with a set size in a dimension, without +trying to make them fit the whole size, this is useful to have a set of +chilgren of the same predefined size, side to side. - .. only:: html +**AnchorLayout**: +A simple layout only caring about children position, allows to stick the +children to a position relative to a border of the layout. +`size_hint` not honored. - .. image:: ../images/gridlayout.gif - :align: right - :width: 200 +**FloatLayout**: +Allow to place children to arbitrary places and size, either absolute or +relative to the layout size. Default size_hint (1, 1) will make everychildren +the same size as the whole layout, so you probably want to change this value +if you have more than one child. You can set size_hint to (None, None) to use +absolute size with `size`, this widget honors `pos_hint` too, which as a dict +allowing to set position relatively to layout position. - .. only:: latex - - .. image:: ../images/gridlayout.png - :align: right - :width: 200 - - **StackLayout**: - Arranges widgets side-by-side, but with a set sizes and without - trying to make them fit the whole size. This is useful if you want a set of - children of some predefined size, side by side. - - `pos_hint` is not honored. - - .. only:: html - - .. image:: ../images/stacklayout.gif - :align: right - :width: 200 - - .. only:: latex - - .. image:: ../images/stacklayout.png - :align: right - :width: 200 - - **AnchorLayout**: - A simple layout which only cares about its childrens position. It allows you to - place children relative to the border of the layout. - - `size_hint` is not honored. - - .. only:: html - - .. image:: ../images/anchorlayout.gif - :align: right - :width: 200 - - .. only:: latex - - .. image:: ../images/anchorlayout.png - :align: right - :width: 200 - - **FloatLayout**: - Allows you to place children in arbitrary places and with arbitrary sizes, using - either absolute or relative coordinates. The default `size_hint` is `(1, 1)`, and will - make the child the same size as the whole layout, so you will probably want to change - this value if you have more than one child. You can set the `size_hint` to `(None, None)` - to use absolute size with `size`. This widget honors `pos_hint`, which is a dict - allowing you to set the position relative to the layouts position. - - .. only:: html - - .. image:: ../images/floatlayout.gif - :align: right - :width: 300 - - .. only:: latex - - .. image:: ../images/floatlayout.png - :align: right - :width: 300 - - **RelativeLayout**: - Behaves just like FloatLayout, except the childrens `pos` is relative to the layout - position, not the screen. +**RelativeLayout**: +Behave just like FloatLayout, except children pos is relative to layout +position, not screen. Look at the documentation of the various Layouts to get a in-depth understanding of how they work with |size_hint| and |pos_hint|: @@ -357,6 +309,162 @@ This code should give us something that looks like this: You can experiment further with |pos_hint| by changing the values to understand the effect they have on the widgets position. +Adding a Background to a Layout +------------------------------- + +One of the frequently asked questions about layouts is: + + "How to add a background image/color/video/... to a Layout" + +Layouts by their nature have no visual representation, i.e. they have no canvas +instructions by default. However you can add instructions to the Layouts canvas. + +To add a color to the background of a **layouts Instance** + +In Python:: + + with layout_instance.canvas.before: + Color(rgba(0, 1, 0, 1)) # green; colors range from 0-1 instead of 0-255 + self.rect = Rectangle( + size=layout_instance.size, + pos=layout_instance.pos) + +Unfortunately this will only draw a rectangle at the layouts initial position +and size.To make sure the rect is drawn inside the layout if layout size/pos +changes we need to listen to any changes and update the Rectangles size and pos +like so:: + + # listen to size and position changes + layout_instance.bind( + size=self._update_rect, + pos=self._update_rect) + + ... + def _update_rect(self, instance, value): + self.rect.pos = instance.pos + self.rect.size = instance.size + +In kv: + +.. code-block:: kv + + ... + ... + FloatLayout: + canvas.before: + Color: + rgba: 0, 1, 0, 1 + Rectangle: + # self here refers to the widget i.e BoxLayout + pos: self.pos + size: self.size + +That's it the binding is implicit. kv language in the last two lines updates the +values |pos| and |size| of the rectangle when the |pos| of the |FloatLayout| +changes. QED. + +Now Let's put the snippets above into the shell of Kivy App. +Pure Python way:: + + from kivy.app import App + from kivy.graphics import Color, Rectangle + from kivy.uix.floatlayout import FloatLayout + from kivy.uix.button import Button + + + class RootWidget(FloatLayout): + + def __init__(self, **kwargs): + # make sure we aren't overriding any important functionality + super(RootWidget, self).__init__(**kwargs) + + with self.canvas.before: + Color(0, 1, 0, 1) # green; colors range from 0-1 instead of 0-255 + self.rect = Rectangle( + size=self.size, + pos=self.pos) + + # let's add a Widgetto this layout + self.add_widget( + Button( text="Hello World", + size_hint= (.5, .5), + pos_hint={'center_x':.5, + 'center_y':.5})) + + self.bind( + size=self._update_rect, + pos=self._update_rect) + + def _update_rect(self, instance, value): + self.rect.pos = instance.pos + self.rect.size = instance.size + + + class MainApp(App): + + def build(self): + return RootWidget() + + if __name__ == '__main__': + MainApp().run() + +Using KV Language:: + + from kivy.app import App + from kivy.uix.floatlayout import FloatLayout + from kivy.lang import Builder + + + Builder.load_string(''' + + canvas.before: + Color: + rgba: 0, 1, 0, 1 + Rectangle: + # self here refers to the widget i.e BoxLayout + pos: self.pos + size: self.size + Button: + text: 'Hello World!!' + size_hint: .5, .5 + pos_hint: {'center_x':.5, 'center_y': .5} + ''') + + class RootWidget(FloatLayout): + pass + + + class MainApp(App): + + def build(self): + return RootWidget() + + if __name__ == '__main__': + MainApp().run() + +Isn't this a lot simpler? + +Both of the Apps should look something like this + +.. image:: images/layout_background.png + +To add a color to the background of a **custom layouts rule/class** + +To add a color to the background of a **layout globally** + +Now Let's have some fun and add a **Image to the background** + +a bit Advanced Topics:: + + How about a **Animated background**? + + Blitt custom data to the background + +Nesting Layouts +--------------- + +Yes! not only can you nest Layouts, it is actually quite fun to seee how extensible nesting Layouts is + Size and position metrics -------------------------