Sideband/sbapp/kivymd/uix/behaviors/elevation.py

899 lines
27 KiB
Python
Raw Normal View History

2022-07-07 16:16:10 -04:00
"""
Behaviors/Elevation
===================
.. seealso::
`Material Design spec, Elevation <https://material.io/design/environment/elevation.html>`_
.. rubric:: Elevation is the relative distance between two surfaces along the z-axis.
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/elevation-previous.png
:align: center
2022-10-08 11:17:59 -04:00
To create an elevation effect, use the :class:`~CommonElevationBehavior` class.
For example, let's create a button with a rectangular elevation effect:
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. tabs::
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. tab:: Declarative style with KV
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivy.lang import Builder
from kivy.uix.behaviors import ButtonBehavior
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
from kivymd.uix.behaviors import (
RectangularRippleBehavior,
BackgroundColorBehavior,
CommonElevationBehavior,
)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
KV = '''
<RectangularElevationButton>
size_hint: None, None
size: "250dp", "50dp"
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
MDScreen:
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
# With elevation effect
RectangularElevationButton:
pos_hint: {"center_x": .5, "center_y": .6}
elevation: 4.5
shadow_offset: 0, 6
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
# Without elevation effect
RectangularElevationButton:
pos_hint: {"center_x": .5, "center_y": .4}
'''
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class RectangularElevationButton(
RectangularRippleBehavior,
CommonElevationBehavior,
ButtonBehavior,
BackgroundColorBehavior,
):
def __init__(self, **kwargs):
super().__init__(**kwargs)
self.md_bg_color = "red"
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class Example(MDApp):
def build(self):
return Builder.load_string(KV)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Example().run()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. tab:: Declarative python style
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivy.uix.behaviors import ButtonBehavior
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
from kivymd.uix.behaviors import (
RectangularRippleBehavior,
BackgroundColorBehavior,
CommonElevationBehavior,
)
from kivymd.uix.screen import MDScreen
2022-10-08 11:17:59 -04:00
class RectangularElevationButton(
RectangularRippleBehavior,
CommonElevationBehavior,
ButtonBehavior,
BackgroundColorBehavior,
):
def __init__(self, **kwargs):
super().__init__(**kwargs)
self.md_bg_color = "red"
self.size_hint = (None, None)
self.size = ("250dp", "50dp")
2022-10-08 11:17:59 -04:00
class Example(MDApp):
def build(self):
return (
MDScreen(
RectangularElevationButton(
pos_hint={"center_x": .5, "center_y": .6},
elevation=4.5,
shadow_offset=(0, 6),
),
RectangularElevationButton(
pos_hint={"center_x": .5, "center_y": .4},
),
)
)
2022-10-08 11:17:59 -04:00
Example().run()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/rectangular-elevation-effect.png
2022-07-07 16:16:10 -04:00
:align: center
2022-10-08 11:17:59 -04:00
.. warning::
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
If before the KivyMD 1.1.0 library version you used the elevation property
with an average value of `12` for the shadow, then starting with the KivyMD
1.1.0 library version, the average value of the elevation property will be
somewhere `4`.
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Similarly, create a circular button:
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. tabs::
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. tab:: Declarative style with KV
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivy.lang import Builder
from kivy.uix.behaviors import ButtonBehavior
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
from kivymd.uix.behaviors import CircularRippleBehavior, CommonElevationBehavior
from kivymd.uix.floatlayout import MDFloatLayout
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
KV = '''
<CircularElevationButton>
size_hint: None, None
size: "100dp", "100dp"
radius: self.size[0] / 2
shadow_radius: self.radius[0]
md_bg_color: "red"
MDIcon:
icon: "hand-heart"
halign: "center"
valign: "center"
pos_hint: {"center_x": .5, "center_y": .5}
size: root.size
pos: root.pos
font_size: root.size[0] * .6
theme_text_color: "Custom"
text_color: "white"
MDScreen:
CircularElevationButton:
pos_hint: {"center_x": .5, "center_y": .6}
elevation: 4
'''
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class CircularElevationButton(
CommonElevationBehavior,
CircularRippleBehavior,
ButtonBehavior,
MDFloatLayout,
):
pass
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class Example(MDApp):
def build(self):
return Builder.load_string(KV)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Example().run()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. tab:: Declarative python style
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-10-08 11:17:59 -04:00
from kivy.metrics import dp
from kivy.uix.behaviors import ButtonBehavior
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
from kivymd.uix.behaviors import CircularRippleBehavior, CommonElevationBehavior
from kivymd.uix.floatlayout import MDFloatLayout
from kivymd.uix.label import MDIcon
from kivymd.uix.screen import MDScreen
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class CircularElevationButton(
CommonElevationBehavior,
CircularRippleBehavior,
ButtonBehavior,
MDFloatLayout,
):
def __init__(self, **kwargs):
super().__init__(**kwargs)
self.size_hint = (None, None)
self.size = (dp(100), dp(100))
self.radius = dp(100) / 2
self.shadow_radius = dp(100) / 2
self.md_bg_color = "red"
self.add_widget(
MDIcon(
icon="hand-heart",
halign="center",
valign="center",
pos_hint={"center_x": .5, "center_y": .5},
size=self.size,
theme_text_color="Custom",
text_color="white",
font_size=self.size[0] * 0.6,
)
)
2022-10-08 11:17:59 -04:00
class Example(MDApp):
def build(self):
return (
MDScreen(
CircularElevationButton(
pos_hint={"center_x": .5, "center_y": .5},
elevation=4,
)
)
)
2022-10-08 11:17:59 -04:00
Example().run()
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/circular-elevation-effect.png
2022-07-07 16:16:10 -04:00
:align: center
Animating the elevation
-----------------------
2022-10-08 11:17:59 -04:00
.. tabs::
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. tab:: Declarative style with KV
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivy.animation import Animation
from kivy.lang import Builder
from kivy.uix.behaviors import ButtonBehavior
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
from kivymd.uix.behaviors import CommonElevationBehavior, RectangularRippleBehavior
from kivymd.uix.widget import MDWidget
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
KV = '''
MDScreen:
ElevatedWidget:
pos_hint: {'center_x': .5, 'center_y': .5}
size_hint: None, None
size: 100, 100
md_bg_color: 0, 0, 1, 1
elevation: 4
radius: 18
'''
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class ElevatedWidget(
CommonElevationBehavior,
RectangularRippleBehavior,
ButtonBehavior,
MDWidget,
):
_elev = 0 # previous elevation value
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_press(self, *args):
if not self._elev:
self._elev = self.elevation
Animation(elevation=self.elevation + 2, d=0.4).start(self)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_release(self, *args):
Animation.cancel_all(self, "elevation")
Animation(elevation=self._elev, d=0.1).start(self)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class Example(MDApp):
def build(self):
return Builder.load_string(KV)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Example().run()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. tab:: Declarative python style
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivy.animation import Animation
from kivy.uix.behaviors import ButtonBehavior
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
from kivymd.uix.behaviors import CommonElevationBehavior, RectangularRippleBehavior
from kivymd.uix.screen import MDScreen
from kivymd.uix.widget import MDWidget
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class ElevatedWidget(
CommonElevationBehavior,
RectangularRippleBehavior,
ButtonBehavior,
MDWidget,
):
_elev = 0 # previous elevation value
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_press(self, *args):
if not self._elev:
self._elev = self.elevation
Animation(elevation=self.elevation + 2, d=0.4).start(self)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_release(self, *args):
Animation.cancel_all(self, "elevation")
Animation(elevation=self._elev, d=0.1).start(self)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class Example(MDApp):
def build(self):
return (
MDScreen(
ElevatedWidget(
pos_hint={'center_x': .5, 'center_y': .5},
size_hint=(None, None),
size=(100, 100),
md_bg_color="blue",
elevation=4,
radius=18,
)
)
)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Example().run()
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/rectangular-elevation-animation-effect.gif
2022-07-07 16:16:10 -04:00
:align: center
"""
2022-10-08 11:17:59 -04:00
from __future__ import annotations
2022-07-07 16:16:10 -04:00
__all__ = (
"CommonElevationBehavior",
"RectangularElevationBehavior",
"CircularElevationBehavior",
"RoundedRectangularElevationBehavior",
"FakeRectangularElevationBehavior",
"FakeCircularElevationBehavior",
)
2022-10-08 11:17:59 -04:00
import os
2022-07-07 16:16:10 -04:00
from kivy import Logger
from kivy.clock import Clock
2022-10-08 11:17:59 -04:00
from kivy.core.window import Window
from kivy.graphics import RenderContext, RoundedRectangle
2022-07-07 16:16:10 -04:00
from kivy.properties import (
AliasProperty,
BooleanProperty,
BoundedNumericProperty,
2022-10-08 11:17:59 -04:00
ColorProperty,
2022-07-07 16:16:10 -04:00
ListProperty,
NumericProperty,
ObjectProperty,
VariableListProperty,
)
from kivy.uix.widget import Widget
2022-10-08 11:17:59 -04:00
from kivymd import glsl_path
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
# FIXME: Add shadow manipulation with canvas instructions such as
# PushMatrix and PopMatrix.
2022-07-07 16:16:10 -04:00
class CommonElevationBehavior(Widget):
"""Common base class for rectangular and circular elevation behavior."""
elevation = BoundedNumericProperty(0, min=0, errorvalue=0)
"""
Elevation of the widget.
:attr:`elevation` is an :class:`~kivy.properties.BoundedNumericProperty`
and defaults to `0`.
"""
2022-10-08 11:17:59 -04:00
shadow_radius = VariableListProperty([0], length=4)
2022-07-07 16:16:10 -04:00
"""
2022-10-08 11:17:59 -04:00
Radius of the corners of the shadow.
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. versionadded:: 1.1.0
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
You don't have to use this parameter.
The radius of the elevation effect is calculated automatically one way
or another based on the radius of the parent widget, for example:
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivy.lang import Builder
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
KV = '''
MDScreen:
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
MDCard:
radius: 12, 46, 12, 46
size_hint: .5, .3
pos_hint: {"center_x": .5, "center_y": .5}
elevation: 4
shadow_softness: 8
shadow_offset: (-2, 2)
'''
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class Test(MDApp):
def build(self):
return Builder.load_string(KV)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Test().run()
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-radius.png
:align: center
2022-10-08 11:17:59 -04:00
.. note::
However, if you want to use this parameter, remember that the angle
values for the radius of the Kivy widgets and the radius for the shader
are different.
2022-07-07 16:16:10 -04:00
.. code-block:: python
2022-10-08 11:17:59 -04:00
shadow_radius = ['top-right', 'bot-right', 'top-left', 'bot-left']
kivy_radius = ['top-left', 'top-right', 'bottom-right', 'bottom-left']
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
:attr:`shadow_radius` is an :class:`~kivy.properties.VariableListProperty`
2022-07-07 16:16:10 -04:00
and defaults to `[0, 0, 0, 0]`.
"""
2022-10-08 11:17:59 -04:00
shadow_softness = NumericProperty(12)
2022-07-07 16:16:10 -04:00
"""
2022-10-08 11:17:59 -04:00
Softness of the shadow.
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. versionadded:: 1.1.0
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivy.lang import Builder
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
from kivymd.uix.behaviors import BackgroundColorBehavior, CommonElevationBehavior
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
KV = '''
<RectangularElevationButton>
size_hint: None, None
size: "250dp", "50dp"
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
MDScreen:
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
RectangularElevationButton:
pos_hint: {"center_x": .5, "center_y": .6}
elevation: 6
shadow_softness: 6
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
RectangularElevationButton:
pos_hint: {"center_x": .5, "center_y": .4}
elevation: 6
shadow_softness: 12
'''
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class RectangularElevationButton(CommonElevationBehavior, BackgroundColorBehavior):
md_bg_color = [0, 0, 1, 1]
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class Example(MDApp):
def build(self):
return Builder.load_string(KV)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Example().run()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-softness.png
:align: center
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
:attr:`shadow_softness` is an :class:`~kivy.properties.NumericProperty`
and defaults to `12`.
2022-07-07 16:16:10 -04:00
"""
2022-10-08 11:17:59 -04:00
shadow_offset = ListProperty((0, 2))
2022-07-07 16:16:10 -04:00
"""
2022-10-08 11:17:59 -04:00
Offset of the shadow.
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. versionadded:: 1.1.0
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivy.lang import Builder
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
from kivymd.app import MDApp
from kivymd.uix.behaviors import BackgroundColorBehavior, CommonElevationBehavior
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
KV = '''
<RectangularElevationButton>
size_hint: None, None
size: "100dp", "100dp"
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
MDScreen:
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
RectangularElevationButton:
pos_hint: {"center_x": .5, "center_y": .5}
elevation: 6
shadow_radius: 18
shadow_softness: 24
shadow_offset: 12, 12
'''
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class RectangularElevationButton(CommonElevationBehavior, BackgroundColorBehavior):
md_bg_color = [0, 0, 1, 1]
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
class Example(MDApp):
def build(self):
return Builder.load_string(KV)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Example().run()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-offset-1.png
:align: center
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: kv
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
RectangularElevationButton:
shadow_offset: -12, 12
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-offset-2.png
:align: center
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: kv
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
RectangularElevationButton:
shadow_offset: -12, -12
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-offset-3.png
:align: center
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: kv
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
RectangularElevationButton:
shadow_offset: 12, -12
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-offset-4.png
:align: center
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
:attr:`shadow_offset` is an :class:`~kivy.properties.ListProperty`
and defaults to `(0, 2)`.
2022-07-07 16:16:10 -04:00
"""
2022-10-08 11:17:59 -04:00
shadow_color = ColorProperty([0, 0, 0, 0.6])
2022-07-07 16:16:10 -04:00
"""
2022-10-08 11:17:59 -04:00
Offset of the shadow.
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. versionadded:: 1.1.0
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. code-block:: python
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
RectangularElevationButton:
shadow_color: 0, 0, 1, .8
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-color.png
:align: center
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
:attr:`shadow_color` is an :class:`~kivy.properties.ColorProperty`
and defaults to `[0.4, 0.4, 0.4, 0.8]`.
"""
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
_transition_ref = ObjectProperty()
_has_relative_position = BooleanProperty(defaultvalue=False)
_elevation = 0
_shadow_color = [0.0, 0.0, 0.0, 0.0]
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def _get_window_pos(self, *args):
window_pos = self.to_window(*self.pos)
# To list, so it can be compared to self.pos directly.
return [window_pos[0], window_pos[1]]
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def _set_window_pos(self, value):
self.window_pos = value
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
window_pos = AliasProperty(_get_window_pos, _set_window_pos)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def __init__(self, **kwargs):
super().__init__(**kwargs)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
with self.canvas.before:
self.context = RenderContext(use_parent_projection=True)
with self.context:
self.rect = RoundedRectangle(pos=self.pos, size=self.size)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
self.after_init()
2022-10-02 11:16:59 -04:00
2022-10-08 11:17:59 -04:00
def after_init(self, *args):
Clock.schedule_once(self.check_for_relative_behavior)
Clock.schedule_once(self.set_shader_string)
Clock.schedule_once(lambda x: self.on_elevation(self, self.elevation))
self.on_pos()
def check_for_relative_behavior(self, *args) -> None:
"""
Checks if the widget has relative properties and if necessary
binds Window.on_draw and screen events to fix behavior
"""
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
if self.pos != self.window_pos:
self._has_relative_position = True
# Loops to check if its inside screenmanager or bottom_navigation.
widget = self
while True:
# Checks if has screen event function
# works for Screen and MDTab objects.
if hasattr(widget, "on_pre_enter"):
widget.bind(on_pre_enter=self.apply_correction)
widget.bind(on_pre_leave=self.apply_correction)
widget.bind(on_enter=self.reset_correction)
widget.bind(on_leave=self.reset_correction)
self._has_relative_position = True
# Save refs to objects with transition property.
if hasattr(widget, "header"): # specific to bottom_nav
self._transition_ref = widget.header.panel
elif hasattr(widget, "manager"): # specific to screen
if widget.manager: # manager cant be None
self._transition_ref = widget.manager
break
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
elif widget.parent and str(widget) != str(widget.parent):
widget = widget.parent
else:
break
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
if self._has_relative_position:
Window.bind(on_draw=self.update_window_position)
def apply_correction(self, *args):
if self._transition_ref:
transition = str(self._transition_ref.transition)
# Slide and Card transitions only need _has_relative_pos to be
# always on.
if (
"SlideTransition" in transition
or "CardTransition" in transition
):
self.context.use_parent_modelview = False
else:
self.context.use_parent_modelview = True
def reset_correction(self, *args):
self.context.use_parent_modelview = False
self.update_window_position()
def get_shader_string(self) -> str:
shader_string = ""
for name_file in ["header.frag", "elevation.frag", "main.frag"]:
with open(
os.path.join(glsl_path, "elevation", name_file),
encoding="utf-8",
) as file:
shader_string += f"{file.read()}\n\n"
return shader_string
def set_shader_string(self, *args) -> None:
self.context["shadow_radius"] = list(map(float, self.shadow_radius))
self.context["shadow_softness"] = float(self.shadow_softness)
self.context["shadow_color"] = list(map(float, self.shadow_color))[
:-1
] + [float(self.opacity)]
self.context["pos"] = list(map(float, self.rect.pos))
self.context.shader.fs = self.get_shader_string()
def update_resolution(self) -> None:
self.context["resolution"] = (*self.rect.size, *self.rect.pos)
def on_shadow_color(self, instance, value) -> None:
def on_shadow_color(*args):
self._shadow_color = list(map(float, value))[:-1] + [
float(self.opacity) if not self.disabled else 0
]
self.context["shadow_color"] = self._shadow_color
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Clock.schedule_once(on_shadow_color)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_shadow_radius(self, instance, value) -> None:
def on_shadow_radius(*args):
if hasattr(self, "context"):
self.context["shadow_radius"] = list(map(float, value))
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Clock.schedule_once(on_shadow_radius)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_shadow_softness(self, instance, value) -> None:
def on_shadow_softness(*args):
if hasattr(self, "context"):
self.context["shadow_softness"] = float(value)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Clock.schedule_once(on_shadow_softness)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_elevation(self, instance, value) -> None:
def on_elevation(*args):
if hasattr(self, "context"):
self._elevation = value
self.hide_elevation(
True if (value <= 0 or self.disabled) else False
)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
Clock.schedule_once(on_elevation)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_shadow_offset(self, instance, value) -> None:
self.on_size()
self.on_pos()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def update_window_position(self, *args) -> None:
2022-07-07 16:16:10 -04:00
"""
2022-10-08 11:17:59 -04:00
This function is used only when the widget has relative position
properties.
2022-07-07 16:16:10 -04:00
"""
2022-10-08 11:17:59 -04:00
self.on_pos()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_pos(self, *args) -> None:
if not hasattr(self, "rect"):
2022-07-07 16:16:10 -04:00
return
2022-10-08 11:17:59 -04:00
if (
self._has_relative_position
and not self.context.use_parent_modelview
):
pos = self.window_pos
else:
pos = self.pos
self.rect.pos = [
pos[0]
- ((self.rect.size[0] - self.width) / 2)
- self.shadow_offset[0],
pos[1]
- ((self.rect.size[1] - self.height) / 2)
- self.shadow_offset[1],
]
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
self.context["mouse"] = [self.rect.pos[0], 0.0, 0.0, 0.0]
self.context["pos"] = list(map(float, self.rect.pos))
self.update_resolution()
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def on_size(self, *args) -> None:
if not hasattr(self, "rect"):
return
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
# If the elevation value is 0, set the canvas size to zero.
# Because even with a zero elevation value, the shadow is displayed
# under the widget. This is visible if we change the scale
# of the widget.
width = self.size[0] if self.elevation else 0
height = self.size[1] if self.elevation else 0
self.rect.size = (
width + (self._elevation * self.shadow_softness / 2),
height + (self._elevation * self.shadow_softness / 2),
2022-07-07 16:16:10 -04:00
)
2022-10-08 11:17:59 -04:00
self.context["mouse"] = [self.rect.pos[0], 0.0, 0.0, 0.0]
self.context["size"] = list(map(float, self.rect.size))
self.update_resolution()
2022-10-08 11:17:59 -04:00
def on_opacity(self, instance, value: int | float) -> None:
"""
2022-10-08 11:17:59 -04:00
Adjusts the transparency of the shadow according to the transparency
of the widget.
"""
2022-10-08 11:17:59 -04:00
def on_opacity(*args):
self._shadow_color = list(map(float, self._shadow_color))[:-1] + [
float(value)
]
2022-10-08 11:17:59 -04:00
self.context["shadow_color"] = self._shadow_color
2022-10-08 11:17:59 -04:00
super().on_opacity(instance, value)
Clock.schedule_once(on_opacity)
2022-10-08 11:17:59 -04:00
def on_radius(self, instance, value) -> None:
self.shadow_radius = [value[1], value[2], value[0], value[3]]
2022-10-08 11:17:59 -04:00
def on_disabled(self, instance, value) -> None:
if value:
self._elevation = 0
self.hide_elevation(True)
2022-10-02 11:16:59 -04:00
else:
2022-10-08 11:17:59 -04:00
self.hide_elevation(False)
2022-07-07 16:16:10 -04:00
2022-10-08 11:17:59 -04:00
def hide_elevation(self, hide: bool) -> None:
if hide:
self._elevation = -self.elevation
self._shadow_color = [0.0, 0.0, 0.0, 0.0]
else:
self._elevation = self.elevation
self._shadow_color = self.shadow_color[:-1] + [float(self.opacity)]
2022-10-08 11:17:59 -04:00
self.on_shadow_color(self, self._shadow_color)
self.on_size()
self.on_pos()
2022-07-07 16:16:10 -04:00
class RectangularElevationBehavior(CommonElevationBehavior):
"""
2022-10-08 11:17:59 -04:00
.. deprecated:: 1.1.0
Use :class:`~CommonElevationBehavior` class instead.
2022-07-07 16:16:10 -04:00
"""
def __init__(self, **kwargs):
super().__init__(**kwargs)
2022-10-08 11:17:59 -04:00
Logger.warning(
"KivyMD: "
"The `RectangularElevationBehavior` class has been deprecated. "
"Use the `CommonElevationBehavior` class instead.`"
)
2022-07-07 16:16:10 -04:00
class CircularElevationBehavior(CommonElevationBehavior):
"""
2022-10-08 11:17:59 -04:00
.. deprecated:: 1.1.0
Use :class:`~CommonElevationBehavior` class instead.
2022-07-07 16:16:10 -04:00
"""
def __init__(self, **kwargs):
super().__init__(**kwargs)
2022-10-08 11:17:59 -04:00
Logger.warning(
"KivyMD: "
"The `CircularElevationBehavior` class has been deprecated. "
"Use the `CommonElevationBehavior` class instead.`"
)
2022-07-07 16:16:10 -04:00
class RoundedRectangularElevationBehavior(CommonElevationBehavior):
"""
2022-10-08 11:17:59 -04:00
.. deprecated:: 1.1.0
Use :class:`~CommonElevationBehavior` class instead.
2022-07-07 16:16:10 -04:00
"""
def __init__(self, **kwargs):
super().__init__(**kwargs)
2022-10-08 11:17:59 -04:00
Logger.warning(
"KivyMD: "
"The `RoundedRectangularElevationBehavior` class has been "
"deprecated. Use the `CommonElevationBehavior` class instead.`"
2022-07-07 16:16:10 -04:00
)
class FakeRectangularElevationBehavior(CommonElevationBehavior):
"""
2022-10-08 11:17:59 -04:00
.. deprecated:: 1.1.0
Use :class:`~CommonElevationBehavior` class instead.
2022-07-07 16:16:10 -04:00
"""
def __init__(self, **kwargs):
super().__init__(**kwargs)
2022-10-08 11:17:59 -04:00
Logger.warning(
"KivyMD: "
"The `FakeRectangularElevationBehavior` class has been "
"deprecated. Use the `CommonElevationBehavior` class instead."
)
2022-07-07 16:16:10 -04:00
class FakeCircularElevationBehavior(CommonElevationBehavior):
"""
2022-10-08 11:17:59 -04:00
.. deprecated:: 1.1.0
Use :class:`~CommonElevationBehavior` class instead.
2022-07-07 16:16:10 -04:00
"""
def __init__(self, **kwargs):
super().__init__(**kwargs)
2022-10-08 11:17:59 -04:00
Logger.warning(
"KivyMD: "
"The `FakeCircularElevationBehavior` class has been deprecated. "
"Use the `CommonElevationBehavior` class instead."
)