From 4b619f385d3d0199b9749e5d288523e40c23c933 Mon Sep 17 00:00:00 2001 From: Mark Qvist Date: Sat, 8 Oct 2022 17:17:59 +0200 Subject: [PATCH] Updated KivyMD --- sbapp/kivymd/__init__.py | 3 + sbapp/kivymd/app.py | 26 +- sbapp/kivymd/color_definitions.py | 4 +- .../kivymd/data/glsl/elevation/elevation.frag | 51 + sbapp/kivymd/data/glsl/elevation/header.frag | 10 + sbapp/kivymd/data/glsl/elevation/main.frag | 10 + sbapp/kivymd/images/folder.png | Bin 29440 -> 2311 bytes sbapp/kivymd/images/quad_shadow-0.png | Bin 31415 -> 0 bytes sbapp/kivymd/images/quad_shadow-1.png | Bin 31667 -> 0 bytes sbapp/kivymd/images/quad_shadow-2.png | Bin 21912 -> 0 bytes sbapp/kivymd/images/quad_shadow.atlas | 1 - sbapp/kivymd/images/rec_shadow-0.png | Bin 46593 -> 0 bytes sbapp/kivymd/images/rec_shadow-1.png | Bin 43957 -> 0 bytes sbapp/kivymd/images/rec_shadow.atlas | 1 - sbapp/kivymd/images/rec_st_shadow-0.png | Bin 30721 -> 0 bytes sbapp/kivymd/images/rec_st_shadow-1.png | Bin 32265 -> 0 bytes sbapp/kivymd/images/rec_st_shadow-2.png | Bin 28526 -> 0 bytes sbapp/kivymd/images/rec_st_shadow.atlas | 1 - sbapp/kivymd/images/round_shadow-0.png | Bin 39635 -> 0 bytes sbapp/kivymd/images/round_shadow-1.png | Bin 40767 -> 0 bytes sbapp/kivymd/images/round_shadow-2.png | Bin 26510 -> 0 bytes sbapp/kivymd/images/round_shadow.atlas | 1 - .../pyinstaller/test_pyinstaller_packaging.py | 9 +- sbapp/kivymd/tests/test_create_project.py | 7 +- sbapp/kivymd/theming.py | 206 +- .../packaging/pyinstaller/hook-kivymd.py | 12 + sbapp/kivymd/uix/backdrop/backdrop.kv | 4 +- sbapp/kivymd/uix/backdrop/backdrop.py | 3 +- sbapp/kivymd/uix/banner/banner.py | 5 +- sbapp/kivymd/uix/behaviors/__init__.py | 6 +- .../uix/behaviors/backgroundcolor_behavior.py | 56 +- ...ve_bahavior.py => declarative_behavior.py} | 0 sbapp/kivymd/uix/behaviors/elevation.py | 2099 ++++++----------- sbapp/kivymd/uix/behaviors/ripple_behavior.py | 2 +- sbapp/kivymd/uix/behaviors/rotate_behavior.py | 133 ++ sbapp/kivymd/uix/behaviors/scale_behavior.py | 156 ++ .../kivymd/uix/behaviors/stencil_behavior.py | 134 ++ sbapp/kivymd/uix/behaviors/toggle_behavior.py | 134 +- .../uix/bottomnavigation/bottomnavigation.kv | 7 +- .../uix/bottomnavigation/bottomnavigation.py | 217 +- sbapp/kivymd/uix/bottomsheet/bottomsheet.py | 6 +- sbapp/kivymd/uix/boxlayout.py | 4 +- sbapp/kivymd/uix/button/__init__.py | 1 + sbapp/kivymd/uix/button/button.kv | 99 +- sbapp/kivymd/uix/button/button.py | 987 +++++--- sbapp/kivymd/uix/card/card.kv | 11 - sbapp/kivymd/uix/card/card.py | 87 +- sbapp/kivymd/uix/chip/chip.py | 16 +- sbapp/kivymd/uix/datatables/datatables.kv | 19 +- sbapp/kivymd/uix/datatables/datatables.py | 176 +- sbapp/kivymd/uix/dialog/dialog.kv | 7 +- sbapp/kivymd/uix/dialog/dialog.py | 238 +- sbapp/kivymd/uix/dropdownitem/dropdownitem.kv | 5 +- sbapp/kivymd/uix/dropdownitem/dropdownitem.py | 15 +- sbapp/kivymd/uix/filemanager/filemanager.kv | 36 +- sbapp/kivymd/uix/filemanager/filemanager.py | 329 ++- sbapp/kivymd/uix/fitimage/fitimage.py | 4 +- sbapp/kivymd/uix/gridlayout.py | 5 +- sbapp/kivymd/uix/hero.py | 142 +- sbapp/kivymd/uix/imagelist/imagelist.py | 5 +- sbapp/kivymd/uix/label/label.kv | 10 +- sbapp/kivymd/uix/label/label.py | 72 +- sbapp/kivymd/uix/list/list.py | 825 +++++-- sbapp/kivymd/uix/menu/menu.kv | 6 +- sbapp/kivymd/uix/menu/menu.py | 6 +- .../uix/navigationdrawer/navigationdrawer.py | 19 +- .../uix/navigationrail/navigationrail.py | 656 ++++-- .../uix/pickers/datepicker/datepicker.kv | 52 +- .../uix/pickers/datepicker/datepicker.py | 654 ++--- .../uix/pickers/timepicker/timepicker.py | 115 +- .../kivymd/uix/refreshlayout/refreshlayout.py | 2 +- sbapp/kivymd/uix/screen.py | 25 +- sbapp/kivymd/uix/screenmanager.py | 77 +- .../uix/segmentedcontrol/segmentedcontrol.kv | 5 +- .../uix/segmentedcontrol/segmentedcontrol.py | 133 +- .../uix/selectioncontrol/selectioncontrol.kv | 2 +- .../uix/selectioncontrol/selectioncontrol.py | 15 +- sbapp/kivymd/uix/slider/slider.kv | 12 +- sbapp/kivymd/uix/slider/slider.py | 2 +- sbapp/kivymd/uix/sliverappbar/sliverappbar.py | 19 +- sbapp/kivymd/uix/snackbar/snackbar.kv | 2 +- sbapp/kivymd/uix/snackbar/snackbar.py | 3 +- sbapp/kivymd/uix/swiper/swiper.py | 8 +- sbapp/kivymd/uix/tab/tab.kv | 4 + sbapp/kivymd/uix/tab/tab.py | 53 +- sbapp/kivymd/uix/taptargetview.py | 172 +- .../templates/rotatewidget/rotatewidget.kv | 9 - .../templates/rotatewidget/rotatewidget.py | 130 +- .../uix/templates/scalewidget/scalewidget.kv | 10 - .../uix/templates/scalewidget/scalewidget.py | 150 +- .../templates/stencilwidget/stencilwidget.kv | 19 - .../templates/stencilwidget/stencilwidget.py | 118 +- sbapp/kivymd/uix/textfield/textfield.kv | 3 + sbapp/kivymd/uix/textfield/textfield.py | 604 ++++- sbapp/kivymd/uix/toolbar/toolbar.py | 25 +- sbapp/kivymd/uix/tooltip/tooltip.py | 2 +- sbapp/kivymd/uix/transition/transition.py | 211 +- sbapp/kivymd/uix/widget.py | 4 +- 98 files changed, 6018 insertions(+), 3706 deletions(-) create mode 100644 sbapp/kivymd/data/glsl/elevation/elevation.frag create mode 100644 sbapp/kivymd/data/glsl/elevation/header.frag create mode 100644 sbapp/kivymd/data/glsl/elevation/main.frag delete mode 100644 sbapp/kivymd/images/quad_shadow-0.png delete mode 100644 sbapp/kivymd/images/quad_shadow-1.png delete mode 100644 sbapp/kivymd/images/quad_shadow-2.png delete mode 100644 sbapp/kivymd/images/quad_shadow.atlas delete mode 100644 sbapp/kivymd/images/rec_shadow-0.png delete mode 100644 sbapp/kivymd/images/rec_shadow-1.png delete mode 100644 sbapp/kivymd/images/rec_shadow.atlas delete mode 100644 sbapp/kivymd/images/rec_st_shadow-0.png delete mode 100644 sbapp/kivymd/images/rec_st_shadow-1.png delete mode 100644 sbapp/kivymd/images/rec_st_shadow-2.png delete mode 100644 sbapp/kivymd/images/rec_st_shadow.atlas delete mode 100644 sbapp/kivymd/images/round_shadow-0.png delete mode 100644 sbapp/kivymd/images/round_shadow-1.png delete mode 100644 sbapp/kivymd/images/round_shadow-2.png delete mode 100644 sbapp/kivymd/images/round_shadow.atlas rename sbapp/kivymd/uix/behaviors/{declarative_bahavior.py => declarative_behavior.py} (100%) create mode 100644 sbapp/kivymd/uix/behaviors/rotate_behavior.py create mode 100644 sbapp/kivymd/uix/behaviors/scale_behavior.py create mode 100644 sbapp/kivymd/uix/behaviors/stencil_behavior.py delete mode 100644 sbapp/kivymd/uix/templates/rotatewidget/rotatewidget.kv delete mode 100644 sbapp/kivymd/uix/templates/scalewidget/scalewidget.kv delete mode 100644 sbapp/kivymd/uix/templates/stencilwidget/stencilwidget.kv diff --git a/sbapp/kivymd/__init__.py b/sbapp/kivymd/__init__.py index a2d8f8f..fa6cb27 100644 --- a/sbapp/kivymd/__init__.py +++ b/sbapp/kivymd/__init__.py @@ -49,6 +49,9 @@ images_path = os.path.join(path, f"images{os.sep}") uix_path = os.path.join(path, "uix") """Path to uix directory.""" +glsl_path = os.path.join(path, "data", "glsl") +"""Path to glsl directory.""" + _log_message = ( "KivyMD:" + (" Release" if release else "") diff --git a/sbapp/kivymd/app.py b/sbapp/kivymd/app.py index abae5f0..dfe3b27 100644 --- a/sbapp/kivymd/app.py +++ b/sbapp/kivymd/app.py @@ -43,9 +43,10 @@ __all__ = ("MDApp",) import os from kivy.app import App +from kivy.clock import Clock from kivy.lang import Builder from kivy.logger import Logger -from kivy.properties import ObjectProperty +from kivy.properties import ObjectProperty, StringProperty from kivymd.theming import ThemeManager @@ -56,13 +57,16 @@ class FpsMonitoring: def fps_monitor_start(self) -> None: """Adds a monitor to the main application window.""" - from kivy.core.window import Window + def add_monitor(*args): + from kivy.core.window import Window - from kivymd.utils.fpsmonitor import FpsMonitor + from kivymd.utils.fpsmonitor import FpsMonitor - monitor = FpsMonitor() - monitor.start() - Window.add_widget(monitor) + monitor = FpsMonitor() + monitor.start() + Window.add_widget(monitor) + + Clock.schedule_once(add_monitor) class MDApp(App, FpsMonitoring): @@ -71,6 +75,16 @@ class MDApp(App, FpsMonitoring): information. """ + icon = StringProperty("kivymd/images/logo/kivymd-icon-512.png") + """ + See :attr:`~kivy.app.App.icon` attribute for more information. + + .. versionadded:: 1.1.0 + + :attr:`icon` is an :class:`~kivy.properties.StringProperty` + adn default to `kivymd/images/logo/kivymd-icon-512.png`. + """ + theme_cls = ObjectProperty() """ Instance of :class:`~ThemeManager` class. diff --git a/sbapp/kivymd/color_definitions.py b/sbapp/kivymd/color_definitions.py index 47a2edf..e4a7de4 100755 --- a/sbapp/kivymd/color_definitions.py +++ b/sbapp/kivymd/color_definitions.py @@ -412,7 +412,7 @@ To demonstrate the shades of the palette, you can run the following code: self.screen = Factory.Root() for name_tab in colors.keys(): - tab = Tab(text=name_tab) + tab = Tab(title=name_tab) self.screen.ids.android_tabs.add_widget(tab) return self.screen @@ -427,7 +427,7 @@ To demonstrate the shades of the palette, you can run the following code: { "viewclass": "ItemColor", "md_bg_color": colors[tab_text][value_color], - "text": value_color, + "title": value_color, } ) diff --git a/sbapp/kivymd/data/glsl/elevation/elevation.frag b/sbapp/kivymd/data/glsl/elevation/elevation.frag new file mode 100644 index 0000000..03f042a --- /dev/null +++ b/sbapp/kivymd/data/glsl/elevation/elevation.frag @@ -0,0 +1,51 @@ +/* +The shader code has been refactored for the KivyMD library. +You can find the original code of this shaders at the links: + +https://www.shadertoy.com/view/WtdSDs +https://www.shadertoy.com/view/fsdyzB + +Additional thanks to iq for optimizing conditional block for individual +corner radius: +https://iquilezles.org/articles/distfunctions +*/ + +// For lower opengl version + +float custom_smoothstep(float a, float b, float x) { + float t = clamp((x - a) / (b - a), 0.0, 1.0); + return t * t * (3.0 - 2.0 * t); +} + +float roundedBoxSDF(vec2 centerPosition, vec2 size, vec4 radius) { + radius.xy = (centerPosition.x > 0.0) ? radius.xy : radius.zw; + radius.x = (centerPosition.y > 0.0) ? radius.x : radius.y; + + vec2 q = abs(centerPosition) - (size - shadow_softness) + radius.x; + return min(max(q.x, q.y), 0.0) + length(max(q, 0.0)) - radius.x; +} + +void mainImage(out vec4 fragColor, in vec2 fragCoord) { + // Smooth the result (free antialiasing). + float edge0 = 0.0; + float smoothedAlpha = 1.0 - custom_smoothstep(0.0, edge0, 1.0); + // Get the resultant shape. + vec4 quadColor = mix( + vec4( + shadow_color[0], + shadow_color[1], + shadow_color[2], + 0.0 + ), + shadow_color, + smoothedAlpha + ); + // Apply a drop shadow effect. + float shadowDistance = roundedBoxSDF( + fragCoord.xy - mouse.xy - (size / 2.0), size / 2.0, shadow_radius + ); + float shadowAlpha = 1.0 - custom_smoothstep( + -shadow_softness, shadow_softness, shadowDistance + ); + fragColor = mix(quadColor, shadow_color, shadowAlpha - smoothedAlpha); +} diff --git a/sbapp/kivymd/data/glsl/elevation/header.frag b/sbapp/kivymd/data/glsl/elevation/header.frag new file mode 100644 index 0000000..acb3a8a --- /dev/null +++ b/sbapp/kivymd/data/glsl/elevation/header.frag @@ -0,0 +1,10 @@ +#ifdef GL_ES + precision highp float; +#endif + +uniform vec4 resolution; +uniform vec4 mouse; +uniform vec2 size; +uniform vec4 shadow_radius; +uniform float shadow_softness; +uniform vec4 shadow_color; diff --git a/sbapp/kivymd/data/glsl/elevation/main.frag b/sbapp/kivymd/data/glsl/elevation/main.frag new file mode 100644 index 0000000..d9b8712 --- /dev/null +++ b/sbapp/kivymd/data/glsl/elevation/main.frag @@ -0,0 +1,10 @@ +vec2 gfc(in vec4 fc) { + vec2 canvas_pos = resolution.zw; + vec2 uv = fc.xy; + uv.y -= canvas_pos.y; + return uv; +} + +void main(void) { + mainImage(gl_FragColor, gfc(gl_FragCoord)); +} diff --git a/sbapp/kivymd/images/folder.png b/sbapp/kivymd/images/folder.png index 58fed42bf3ef2c8f4ac6cf9c7d40d2153be68324..fe0ed5bab2fe2fa84524331f1b7cb9c4f1c0b6ab 100644 GIT binary patch literal 2311 zcmb_eeKZq#AOCHJ5wW-_MecTMjbi8WT!cnnUOVQP6ii>O#+VY5NyHhcu zDR$9wyLk(n)4W7XdCe0O4^k$hV&3+w|LVW{oO93b{Lc4tKHuN(d(Q83zArz9mj_W> z(@+xtpzZ2{Cj$UcLI{AXDNWFY`;khc7VbvG1I6c6(tIyp$!Ns7oTLI^j{3YIi9FmT zC8-|o>h7#Qqp@4-+anYwqw`8?cf7NI{Ew8Fn27W7;KvIQVet_^nIuHTpEV)6x_g~S z+im~=oZ^ai@<|+(j{#<^_}(GA{`BStk2P{9#y$oL z#KeGa9sijB#ZFT63#HhaO@^K=@6x3M00nSh0{#x5b#8d>c0GcDYlcQ}=d^ zUzM6Vqo}FIEoPVbkuebx0fXIk+D@Qmd*c`K6|@)QS4EE{w)WR#o2T(UYu(0D64CZ@v^ZAq{|#|?KzpS(CMq2^TDTV zQU^@1GzeDiOg~TqmM-DPb3E7tV|zuRaB<4Zx9v{lA4Z|6(21v-+S>G|IEVJ%8((TfhtBhYEM58Sq|<9Y;Z`Q>ExtJjxFqA1-OOed@@n{hkTa}guWTj zI%uNS3vtvAe-J{cFaY-%{F;^>Cqgbeg9#Hu#j7k?$Iz3Wk)(Rt{y{o$>=7g2z^d=i z(VZTkq!pnH?$6m?$?&k-c|=;j);t9T?mtw)l_v^-u&GeT-`*7T>2Hz=##yDeR;QgH zbTz7FAFMD0Q*o?y|f1%QS726lx81Q|gdo z2*uvKOuI*#BeBWWgB!?>d@6LVtHI3=0vB@Gs-AOj|R#bnjg-za^i0Ih|sS93qo$FI?3J z?a&A7+>8aM^@d9-5A+dk2w69|3rz{iUdh?(k)T*ig{M)0Rd_}|*#~ox&-$&>^k~xR z{f(%VN^PhxO-I=Gp4e&I(mZBCnF%^yzlRLYzG0U5^7@rYARCG>gFs5+%&z#El@2od z_fInai2#O?vFsAbjRd7)`%y?>WeFXNwPb<5v)yMn4<&gAfz`mVXLJ7wf`Vjhy{Yc_ zn*+eg3UHMb_eG2P0{%U13;t#FuR~JHW8mBhyr?#;;9C`;r}wrE4u zFf3XWSqrGABc@+$@$ee;tY;C;Bj5D-NxiVF@mHzc_$kS02gmb{aQy3FRr9C;n890kZh1$oG(r7O(Iz_Shn4e5 zIN%;V?QoL{q+=HO>p5ChF{%%eXk#u{4g;)3w}PP-`f+Q`$J`tATgp}-TEr6X{U|Mq z`)tCj=Xz)N$8l()CpbP_qqr=#1z7nxL%;_GO$my1xlhZ2k00Q_)0wC9YOJYUEkKFWu&t0ZcPNBzg#xl_EiCVoudD@3p=TBs%D#Z TGlJZe{}bTq?18U77JB7>kpHcX literal 29440 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 diff --git a/sbapp/kivymd/images/quad_shadow-0.png b/sbapp/kivymd/images/quad_shadow-0.png deleted file mode 100644 index f847665c42b733c2bfd706fb360aba2b43fea42d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 31415 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 diff --git a/sbapp/kivymd/images/quad_shadow-2.png b/sbapp/kivymd/images/quad_shadow-2.png deleted file mode 100644 index 1949d5e7f2686fdea550b32c28751990ec05d964..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 21912 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

*M9o$_nNY&xp`?`=iTXXj*i`WHA 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 Xpv^y3gVgrO|5*wbx-Uc&lvMgRM07r? diff --git a/sbapp/kivymd/images/quad_shadow.atlas b/sbapp/kivymd/images/quad_shadow.atlas deleted file mode 100644 index 68e0aad..0000000 --- a/sbapp/kivymd/images/quad_shadow.atlas +++ /dev/null @@ -1 +0,0 @@ -{"quad_shadow-1.png": {"20": [2, 136, 128, 128], "21": [132, 136, 128, 128], "22": [262, 136, 128, 128], "23": [2, 6, 128, 128], "19": [132, 266, 128, 128], "18": [2, 266, 128, 128], "1": [262, 266, 128, 128], "3": [262, 6, 128, 128], "2": [132, 6, 128, 128]}, "quad_shadow-0.png": {"11": [262, 266, 128, 128], "10": [132, 266, 128, 128], "13": [132, 136, 128, 128], "12": [2, 136, 128, 128], "15": [2, 6, 128, 128], "14": [262, 136, 128, 128], "17": [262, 6, 128, 128], "16": [132, 6, 128, 128], "0": [2, 266, 128, 128]}, "quad_shadow-2.png": {"5": [132, 266, 128, 128], "4": [2, 266, 128, 128], "7": [2, 136, 128, 128], "6": [262, 266, 128, 128], "9": [262, 136, 128, 128], "8": [132, 136, 128, 128]}} \ No newline at end of file diff --git a/sbapp/kivymd/images/rec_shadow-0.png b/sbapp/kivymd/images/rec_shadow-0.png deleted file mode 100644 index f02b919a2d8a5fc16a59f984c46ec881b5d69cf6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 46593 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�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 Vs`!V~6-HfPW?}VP6KRr*>ly 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 diff --git a/sbapp/kivymd/images/rec_st_shadow-1.png b/sbapp/kivymd/images/rec_st_shadow-1.png deleted file mode 100644 index 759ee65285764c0619509c5b8cb27f972b2ab3fa..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 32265 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 G;{P8x^pl$a diff --git a/sbapp/kivymd/images/rec_st_shadow.atlas b/sbapp/kivymd/images/rec_st_shadow.atlas deleted file mode 100644 index d4c24ab..0000000 --- a/sbapp/kivymd/images/rec_st_shadow.atlas +++ /dev/null @@ -1 +0,0 @@ -{"rec_st_shadow-0.png": {"11": [262, 138, 128, 256], "10": [132, 138, 128, 256], "13": [522, 138, 128, 256], "12": [392, 138, 128, 256], "15": [782, 138, 128, 256], "14": [652, 138, 128, 256], "16": [912, 138, 128, 256], "0": [2, 138, 128, 256]}, "rec_st_shadow-1.png": {"20": [522, 138, 128, 256], "21": [652, 138, 128, 256], "17": [2, 138, 128, 256], "23": [912, 138, 128, 256], "19": [262, 138, 128, 256], "18": [132, 138, 128, 256], "22": [782, 138, 128, 256], "1": [392, 138, 128, 256]}, "rec_st_shadow-2.png": {"3": [132, 138, 128, 256], "2": [2, 138, 128, 256], "5": [392, 138, 128, 256], "4": [262, 138, 128, 256], "7": [652, 138, 128, 256], "6": [522, 138, 128, 256], "9": [912, 138, 128, 256], "8": [782, 138, 128, 256]}} \ No newline at end of file diff --git a/sbapp/kivymd/images/round_shadow-0.png b/sbapp/kivymd/images/round_shadow-0.png deleted file mode 100644 index 26d984051562b2baf7d39cb92639aae94d222bda..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 39635 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 HcaHmiYza5P diff --git a/sbapp/kivymd/images/round_shadow-1.png b/sbapp/kivymd/images/round_shadow-1.png deleted file mode 100644 index d0f4c0fd4a6f9ac07d583e28e27e20a2495c392f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 40767 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

(Ig{HG6T4&}_d3cF!08sm%T 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 diff --git a/sbapp/kivymd/images/round_shadow-2.png b/sbapp/kivymd/images/round_shadow-2.png deleted file mode 100644 index d5feef2c8a5a7bedd94cfce29af5cea57fab701f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 26510 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 diff --git a/sbapp/kivymd/images/round_shadow.atlas b/sbapp/kivymd/images/round_shadow.atlas deleted file mode 100644 index f25016d..0000000 --- a/sbapp/kivymd/images/round_shadow.atlas +++ /dev/null @@ -1 +0,0 @@ -{"round_shadow-1.png": {"20": [2, 136, 128, 128], "21": [132, 136, 128, 128], "22": [262, 136, 128, 128], "23": [2, 6, 128, 128], "19": [132, 266, 128, 128], "18": [2, 266, 128, 128], "1": [262, 266, 128, 128], "3": [262, 6, 128, 128], "2": [132, 6, 128, 128]}, "round_shadow-0.png": {"11": [262, 266, 128, 128], "10": [132, 266, 128, 128], "13": [132, 136, 128, 128], "12": [2, 136, 128, 128], "15": [2, 6, 128, 128], "14": [262, 136, 128, 128], "17": [262, 6, 128, 128], "16": [132, 6, 128, 128], "0": [2, 266, 128, 128]}, "round_shadow-2.png": {"5": [132, 266, 128, 128], "4": [2, 266, 128, 128], "7": [2, 136, 128, 128], "6": [262, 266, 128, 128], "9": [262, 136, 128, 128], "8": [132, 136, 128, 128]}} \ No newline at end of file diff --git a/sbapp/kivymd/tests/pyinstaller/test_pyinstaller_packaging.py b/sbapp/kivymd/tests/pyinstaller/test_pyinstaller_packaging.py index 801f5bf..3b81fdf 100644 --- a/sbapp/kivymd/tests/pyinstaller/test_pyinstaller_packaging.py +++ b/sbapp/kivymd/tests/pyinstaller/test_pyinstaller_packaging.py @@ -35,8 +35,15 @@ assert "Icons" in LabelBase._fonts.keys() # NOQA images = os.listdir(kivymd.images_path) print(images) +assert "logo" in images +assert "alpha_layer.png" in images +assert "black.png" in images +assert "blue.png" in images +assert "red.png" in images +assert "green.png" in images +assert "yellow.png" in images assert "folder.png" in images -assert "rec_shadow.atlas" in images +assert "transparent.png" in images """ ) pyi_main.run( diff --git a/sbapp/kivymd/tests/test_create_project.py b/sbapp/kivymd/tests/test_create_project.py index 3122f1e..31265fa 100644 --- a/sbapp/kivymd/tests/test_create_project.py +++ b/sbapp/kivymd/tests/test_create_project.py @@ -1,14 +1,13 @@ def test_create_project(): import os - import sys os.system( - f"{sys.executable} -m kivymd.tools.patterns.create_project " + f"python3.10 -m kivymd.tools.patterns.create_project " f"MVC " f"{os.path.expanduser('~')} " f"TestProject " - f"{sys.executable} " - f"master " + f"python3.10 " + f"stable " f"--name_screen TestProjectScreen " f"--name_database restdb " f"--use_hotreload yes" diff --git a/sbapp/kivymd/theming.py b/sbapp/kivymd/theming.py index 638ae60..b0fa0da 100755 --- a/sbapp/kivymd/theming.py +++ b/sbapp/kivymd/theming.py @@ -212,9 +212,8 @@ respects, the theming stays as documented. dictionary :attr:`kivymd.color_definition.colors`. """ - +from kivy.animation import Animation from kivy.app import App -from kivy.atlas import Atlas from kivy.clock import Clock from kivy.core.window import Window from kivy.event import EventDispatcher @@ -224,13 +223,13 @@ from kivy.properties import ( BooleanProperty, ColorProperty, DictProperty, + NumericProperty, ObjectProperty, OptionProperty, StringProperty, ) from kivy.utils import get_color_from_hex -from kivymd import images_path from kivymd.color_definitions import colors, hue, palette from kivymd.font_definitions import theme_font_styles from kivymd.material_resources import DEVICE_IOS, DEVICE_TYPE @@ -624,6 +623,152 @@ class ThemeManager(EventDispatcher): and defaults to `'M2'`. """ + theme_style_switch_animation = BooleanProperty(False) + """ + Animate app colors when switching app color scheme ('Dark/light'). + + .. versionadded:: 1.1.0 + + .. tabs:: + + .. tab:: Declarative KV style + + .. code-block:: python + + from kivy.lang import Builder + + from kivymd.app import MDApp + + KV = ''' + MDScreen: + + MDCard: + orientation: "vertical" + padding: 0, 0, 0 , "36dp" + size_hint: .5, .5 + pos_hint: {"center_x": .5, "center_y": .5} + elevation: 4 + shadow_radius: 6 + shadow_offset: 0, 2 + + MDLabel: + text: "Theme style - {}".format(app.theme_cls.theme_style) + halign: "center" + valign: "center" + bold: True + font_style: "H5" + + MDRaisedButton: + text: "Set theme" + on_release: app.switch_theme_style() + pos_hint: {"center_x": .5} + ''' + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style_switch_animation = True + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) + + def switch_theme_style(self): + self.theme_cls.primary_palette = ( + "Orange" if self.theme_cls.primary_palette == "Red" else "Red" + ) + self.theme_cls.theme_style = ( + "Dark" if self.theme_cls.theme_style == "Light" else "Light" + ) + + + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.button import MDRaisedButton + from kivymd.uix.card import MDCard + from kivymd.uix.label import MDLabel + from kivymd.uix.screen import MDScreen + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style_switch_animation = True + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDCard( + MDLabel( + id="label", + text="Theme style - {}".format(self.theme_cls.theme_style), + halign="center", + valign="center", + bold=True, + font_style="H5", + ), + MDRaisedButton( + text="Set theme", + on_release=self.switch_theme_style, + pos_hint={"center_x": 0.5}, + ), + id="card", + orientation="vertical", + padding=(0, 0, 0, "36dp"), + size_hint=(0.5, 0.5), + pos_hint={"center_x": 0.5, "center_y": 0.5}, + elevation=4, + shadow_radius=6, + shadow_offset=(0, 2), + ) + ) + ) + + def switch_theme_style(self, *args): + self.theme_cls.primary_palette = ( + "Orange" if self.theme_cls.primary_palette == "Red" else "Red" + ) + self.theme_cls.theme_style = ( + "Dark" if self.theme_cls.theme_style == "Light" else "Light" + ) + self.root.ids.card.ids.label.text = ( + "Theme style - {}".format(self.theme_cls.theme_style) + ) + + + Example().run() + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/theme-style-switch-animation.gif + :align: center + + :attr:`theme_style_switch_animation` is an :class:`~kivy.properties.BooleanProperty` + and defaults to `False`. + """ + + theme_style_switch_animation_duration = NumericProperty(0.2) + """ + Duration of the animation of switching the color scheme of the application + ("Dark/light"). + + .. versionadded:: 1.1.0 + + .. code-block:: python + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style_switch_animation = True + self.theme_cls.theme_style_switch_animation_duration = 0.8 + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/theme-style-switch-animation-duration.gif + :align: center + + :attr:`theme_style_switch_animation_duration` is an :class:`~kivy.properties.NumericProperty` + and defaults to `0.2`. + """ + theme_style = OptionProperty("Light", options=["Light", "Dark"]) """ App theme style. @@ -1184,14 +1329,22 @@ class ThemeManager(EventDispatcher): ): self.set_clearcolor_by_theme_style(theme_style) - set_clearcolor = BooleanProperty(True) + _set_clearcolor = False def set_clearcolor_by_theme_style(self, theme_style): - if not self.set_clearcolor: - return - Window.clearcolor = get_color_from_hex( - self.colors[theme_style]["Background"] - ) + if self.theme_style_switch_animation and self._set_clearcolor: + Animation( + clearcolor=get_color_from_hex( + self.colors[theme_style]["Background"] + ), + d=self.theme_style_switch_animation_duration, + t="linear", + ).start(Window) + else: + Window.clearcolor = get_color_from_hex( + self.colors[theme_style]["Background"] + ) + self._set_clearcolor = True # Font name, size (sp), always caps, letter spacing (sp). font_styles = DictProperty( @@ -1398,10 +1551,6 @@ class ThemeManager(EventDispatcher): def __init__(self, **kwargs): super().__init__(**kwargs) - self.rec_shadow = Atlas(f"{images_path}rec_shadow.atlas") - self.rec_st_shadow = Atlas(f"{images_path}rec_st_shadow.atlas") - self.quad_shadow = Atlas(f"{images_path}quad_shadow.atlas") - self.round_shadow = Atlas(f"{images_path}round_shadow.atlas") Clock.schedule_once(lambda x: self.on_theme_style(0, self.theme_style)) self._determine_device_orientation(None, Window.size) Window.bind(size=self._determine_device_orientation) @@ -1487,6 +1636,16 @@ class ThemableBehavior(EventDispatcher): """ def __init__(self, **kwargs): + self.unbind_properties = [ + "theme_style", + "material_style", + "device_orientation", + "primary_color", + "primary_palette", + "accent_palette", + "text_color", + ] + if self.theme_cls is not None: pass else: @@ -1507,3 +1666,24 @@ class ThemableBehavior(EventDispatcher): ) self.theme_cls = App.get_running_app().theme_cls super().__init__(**kwargs) + + def dec_disabled(self, *args, **kwargs) -> None: + callabacks = self.theme_cls.get_property_observers("theme_style") + + for callaback in callabacks: + try: + if hasattr(callaback, "proxy") and hasattr( + callaback.proxy, "theme_cls" + ): + for property_name in self.unbind_properties: + self.theme_cls.unbind( + **{ + property_name: getattr( + callaback.proxy, callaback.method_name + ) + } + ) + except ReferenceError: + pass + + super().dec_disabled(*args, **kwargs) diff --git a/sbapp/kivymd/tools/packaging/pyinstaller/hook-kivymd.py b/sbapp/kivymd/tools/packaging/pyinstaller/hook-kivymd.py index d2cb9a5..040d103 100644 --- a/sbapp/kivymd/tools/packaging/pyinstaller/hook-kivymd.py +++ b/sbapp/kivymd/tools/packaging/pyinstaller/hook-kivymd.py @@ -13,6 +13,18 @@ from pathlib import Path import kivymd datas = [ + # Add `.frag` files from the `kivymd/data/glsl/elevation` directory. + ( + str(Path(kivymd.glsl_path).joinpath("elevation")) + os.sep, + str( + Path("kivymd").joinpath( + str(Path(kivymd.glsl_path)).split(str(Path("kivymd")) + os.sep)[ + 1 + ] + + f"{os.sep}elevation" + ) + ), + ), # Add `.ttf` files from the `kivymd/fonts` directory. ( kivymd.fonts_path, diff --git a/sbapp/kivymd/uix/backdrop/backdrop.kv b/sbapp/kivymd/uix/backdrop/backdrop.kv index 372c754..2f56d5b 100644 --- a/sbapp/kivymd/uix/backdrop/backdrop.kv +++ b/sbapp/kivymd/uix/backdrop/backdrop.kv @@ -35,8 +35,8 @@ if not root.front_layer_color \ else root.front_layer_color radius: - [root.radius_left, root.radius_left, - root.radius_right, root.radius_right] + [root.radius_left, root.radius_right, + 0, 0] OneLineListItem: id: header_button diff --git a/sbapp/kivymd/uix/backdrop/backdrop.py b/sbapp/kivymd/uix/backdrop/backdrop.py index 4262d58..604ab13 100644 --- a/sbapp/kivymd/uix/backdrop/backdrop.py +++ b/sbapp/kivymd/uix/backdrop/backdrop.py @@ -202,7 +202,6 @@ from kivy.uix.boxlayout import BoxLayout from kivymd import uix_path from kivymd.theming import ThemableBehavior -from kivymd.uix.behaviors import FakeRectangularElevationBehavior from kivymd.uix.boxlayout import MDBoxLayout from kivymd.uix.card import MDCard from kivymd.uix.floatlayout import MDFloatLayout @@ -528,5 +527,5 @@ class _BackLayer(BoxLayout): pass -class _FrontLayer(MDCard, FakeRectangularElevationBehavior): +class _FrontLayer(MDCard): pass diff --git a/sbapp/kivymd/uix/banner/banner.py b/sbapp/kivymd/uix/banner/banner.py index 7feb5c9..fb87221 100644 --- a/sbapp/kivymd/uix/banner/banner.py +++ b/sbapp/kivymd/uix/banner/banner.py @@ -35,7 +35,7 @@ Usage MDTopAppBar: id: toolbar title: "Example Banners" - elevation: 10 + elevation: 4 pos_hint: {'top': 1} MDBoxLayout: @@ -157,7 +157,6 @@ from kivy.properties import ( from kivy.uix.widget import Widget from kivymd import uix_path -from kivymd.uix.behaviors import FakeRectangularElevationBehavior from kivymd.uix.boxlayout import MDBoxLayout from kivymd.uix.button import MDFlatButton from kivymd.uix.card import MDCard @@ -177,7 +176,7 @@ with open( Builder.load_string(kv_file.read()) -class MDBanner(MDCard, FakeRectangularElevationBehavior): +class MDBanner(MDCard): vertical_pad = NumericProperty(dp(68)) """ Indent the banner at the top of the screen. diff --git a/sbapp/kivymd/uix/behaviors/__init__.py b/sbapp/kivymd/uix/behaviors/__init__.py index 9461c39..e89ed10 100755 --- a/sbapp/kivymd/uix/behaviors/__init__.py +++ b/sbapp/kivymd/uix/behaviors/__init__.py @@ -11,18 +11,20 @@ from .backgroundcolor_behavior import ( ) # flake8: NOQA -from .declarative_bahavior import DeclarativeBehavior +from .declarative_behavior import DeclarativeBehavior from .elevation import ( CircularElevationBehavior, CommonElevationBehavior, FakeCircularElevationBehavior, FakeRectangularElevationBehavior, - ObservableShadow, RectangularElevationBehavior, RoundedRectangularElevationBehavior, ) from .magic_behavior import MagicBehavior from .ripple_behavior import CircularRippleBehavior, RectangularRippleBehavior +from .rotate_behavior import RotateBehavior +from .scale_behavior import ScaleBehavior +from .stencil_behavior import StencilBehavior from .touch_behavior import TouchBehavior from .hover_behavior import HoverBehavior # isort:skip diff --git a/sbapp/kivymd/uix/behaviors/backgroundcolor_behavior.py b/sbapp/kivymd/uix/behaviors/backgroundcolor_behavior.py index d96fd60..4c4649b 100755 --- a/sbapp/kivymd/uix/behaviors/backgroundcolor_behavior.py +++ b/sbapp/kivymd/uix/behaviors/backgroundcolor_behavior.py @@ -7,8 +7,9 @@ Behaviors/Background Color __all__ = ("BackgroundColorBehavior", "SpecificBackgroundColorBehavior") -from typing import List +from typing import List, Union +from kivy.animation import Animation from kivy.lang import Builder from kivy.properties import ( ColorProperty, @@ -24,8 +25,6 @@ from kivy.utils import get_color_from_hex from kivymd.color_definitions import hue, palette, text_colors from kivymd.theming import ThemeManager -from .elevation import CommonElevationBehavior - Builder.load_string( """ #:import RelativeLayout kivy.uix.relativelayout.RelativeLayout @@ -38,7 +37,7 @@ Builder.load_string( angle: self.angle origin: self._background_origin Color: - rgba: self.md_bg_color + rgba: self._md_bg_color RoundedRectangle: group: "Background_instruction" size: self.size @@ -67,7 +66,7 @@ Builder.load_string( ) -class BackgroundColorBehavior(CommonElevationBehavior): +class BackgroundColorBehavior: background = StringProperty() """ Background image path. @@ -153,15 +152,26 @@ class BackgroundColorBehavior(CommonElevationBehavior): _background_x = NumericProperty(0) _background_y = NumericProperty(0) - _background_origin = ReferenceListProperty( - _background_x, - _background_y, - ) + _background_origin = ReferenceListProperty(_background_x, _background_y) + _md_bg_color = ColorProperty([0, 0, 0, 0]) def __init__(self, **kwarg): super().__init__(**kwarg) self.bind(pos=self.update_background_origin) + def on_md_bg_color(self, instance_md_widget, color: Union[list, str]): + if ( + hasattr(self, "theme_cls") + and self.theme_cls.theme_style_switch_animation + ): + Animation( + _md_bg_color=color, + d=self.theme_cls.theme_style_switch_animation_duration, + t="linear", + ).start(self) + else: + self._md_bg_color = color + def update_background_origin( self, instance_md_widget, pos: List[float] ) -> None: @@ -206,12 +216,14 @@ class SpecificBackgroundColorBehavior(BackgroundColorBehavior): super().__init__(**kwargs) if hasattr(self, "theme_cls"): self.theme_cls.bind( - primary_palette=self._update_specific_text_color + primary_palette=self._update_specific_text_color, + accent_palette=self._update_specific_text_color, + theme_style=self._update_specific_text_color, ) - self.theme_cls.bind(accent_palette=self._update_specific_text_color) - self.theme_cls.bind(theme_style=self._update_specific_text_color) - self.bind(background_hue=self._update_specific_text_color) - self.bind(background_palette=self._update_specific_text_color) + self.bind( + background_hue=self._update_specific_text_color, + background_palette=self._update_specific_text_color, + ) self._update_specific_text_color(None, None) def _update_specific_text_color( @@ -234,5 +246,17 @@ class SpecificBackgroundColorBehavior(BackgroundColorBehavior): secondary_color[3] = 0.54 else: secondary_color[3] = 0.7 - self.specific_text_color = color - self.specific_secondary_text_color = secondary_color + + if ( + hasattr(self, "theme_cls") + and self.theme_cls.theme_style_switch_animation + ): + Animation( + specific_text_color=color, + specific_secondary_text_color=secondary_color, + d=self.theme_cls.theme_style_switch_animation_duration, + t="linear", + ).start(self) + else: + self.specific_text_color = color + self.specific_secondary_text_color = secondary_color diff --git a/sbapp/kivymd/uix/behaviors/declarative_bahavior.py b/sbapp/kivymd/uix/behaviors/declarative_behavior.py similarity index 100% rename from sbapp/kivymd/uix/behaviors/declarative_bahavior.py rename to sbapp/kivymd/uix/behaviors/declarative_behavior.py diff --git a/sbapp/kivymd/uix/behaviors/elevation.py b/sbapp/kivymd/uix/behaviors/elevation.py index 584c82b..47b2af0 100755 --- a/sbapp/kivymd/uix/behaviors/elevation.py +++ b/sbapp/kivymd/uix/behaviors/elevation.py @@ -11,419 +11,12 @@ Behaviors/Elevation .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/elevation-previous.png :align: center -There are 5 classes in KivyMD that can simulate shadow: +To create an elevation effect, use the :class:`~CommonElevationBehavior` class. +For example, let's create a button with a rectangular elevation effect: - #. :class:`~FakeRectangularElevationBehavior` - #. :class:`~FakeCircularElevationBehavior` +.. tabs:: - #. :class:`~RectangularElevationBehavior` - #. :class:`~CircularElevationBehavior` - #. :class:`~RoundedRectangularElevationBehavior` - -By default, KivyMD widgets use the elevation behavior implemented in classes -:class:`~FakeRectangularElevationBehavior` and :class:`~FakeCircularElevationBehavior` -for cast shadows. These classes use the old method of rendering shadows and it -doesn't look very aesthetically pleasing. Shadows are harsh, no softness: - -The :class:`~RectangularElevationBehavior`, :class:`~CircularElevationBehavior`, -:class:`~RoundedRectangularElevationBehavior` classes use the new shadow -rendering algorithm, based on textures creation using the `Pillow` library. -It looks very aesthetically pleasing and beautiful. - -.. warning:: Remember that :class:`~RectangularElevationBehavior`, - :class:`~CircularElevationBehavior`, :class:`~RoundedRectangularElevationBehavior` - classes require a lot of resources from the device on which your application will run, - so you should not use these classes on mobile devices. - -.. code-block:: python - - from kivy.lang import Builder - from kivy.uix.widget import Widget - - from kivymd.app import MDApp - from kivymd.uix.card import MDCard - from kivymd.uix.behaviors import RectangularElevationBehavior - from kivymd.uix.boxlayout import MDBoxLayout - - KV = ''' - - adaptive_size: True - orientation: "vertical" - spacing: "36dp" - - - - size_hint: None, None - size: 100, 100 - md_bg_color: 0, 0, 1, 1 - elevation: 36 - pos_hint: {'center_x': .5} - - - MDFloatLayout: - - MDBoxLayout: - adaptive_size: True - pos_hint: {'center_x': .5, 'center_y': .5} - spacing: "56dp" - - Box: - - MDLabel: - text: "Deprecated shadow rendering" - adaptive_size: True - - DeprecatedShadowWidget: - - MDLabel: - text: "Doesn't require a lot of resources" - adaptive_size: True - - Box: - - MDLabel: - text: "New shadow rendering" - adaptive_size: True - - NewShadowWidget: - - MDLabel: - text: "It takes a lot of resources" - adaptive_size: True - ''' - - - class BaseShadowWidget(Widget): - pass - - - class DeprecatedShadowWidget(MDCard, BaseShadowWidget): - '''Deprecated shadow rendering. Doesn't require a lot of resources.''' - - - class NewShadowWidget(RectangularElevationBehavior, BaseShadowWidget, MDBoxLayout): - '''New shadow rendering. It takes a lot of resources.''' - - - class Example(MDApp): - def build(self): - return Builder.load_string(KV) - - - Example().run() - -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/elevation-differential.png - :align: center - - -For example, let's create an button with a rectangular elevation effect: - -.. code-block:: python - - from kivy.lang import Builder - from kivy.uix.behaviors import ButtonBehavior - - from kivymd.app import MDApp - from kivymd.uix.behaviors import ( - RectangularRippleBehavior, - BackgroundColorBehavior, - FakeRectangularElevationBehavior, - ) - - KV = ''' - : - size_hint: None, None - size: "250dp", "50dp" - - - MDScreen: - - # With elevation effect - RectangularElevationButton: - pos_hint: {"center_x": .5, "center_y": .6} - elevation: 18 - - # Without elevation effect - RectangularElevationButton: - pos_hint: {"center_x": .5, "center_y": .4} - ''' - - - class RectangularElevationButton( - RectangularRippleBehavior, - FakeRectangularElevationBehavior, - ButtonBehavior, - BackgroundColorBehavior, - ): - md_bg_color = [0, 0, 1, 1] - - - class Example(MDApp): - def build(self): - return Builder.load_string(KV) - - - Example().run() - -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/rectangular-elevation-effect.gif - :align: center - -Similarly, create a circular button: - -.. code-block:: python - - from kivy.lang import Builder - from kivy.uix.behaviors import ButtonBehavior - - from kivymd.uix.boxlayout import MDBoxLayout - from kivymd.app import MDApp - from kivymd.uix.behaviors import ( - CircularRippleBehavior, - FakeCircularElevationBehavior, - ) - - KV = ''' - : - size_hint: None, None - size: "100dp", "100dp" - radius: self.size[0] / 2 - md_bg_color: 0, 0, 1, 1 - - MDIcon: - icon: "hand-heart" - halign: "center" - valign: "center" - size: root.size - pos: root.pos - font_size: root.size[0] * .6 - theme_text_color: "Custom" - text_color: [1] * 4 - - - MDScreen: - - CircularElevationButton: - pos_hint: {"center_x": .5, "center_y": .6} - elevation: 24 - ''' - - - class CircularElevationButton( - FakeCircularElevationBehavior, - CircularRippleBehavior, - ButtonBehavior, - MDBoxLayout, - ): - pass - - - - class Example(MDApp): - def build(self): - return Builder.load_string(KV) - - - Example().run() - -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/circular-fake-elevation.png - :align: center - -Animating the elevation ------------------------ - -.. code-block:: python - - from kivy.animation import Animation - from kivy.lang import Builder - from kivy.properties import ObjectProperty - from kivy.uix.behaviors import ButtonBehavior - - from kivymd.app import MDApp - from kivymd.theming import ThemableBehavior - from kivymd.uix.behaviors import FakeRectangularElevationBehavior, RectangularRippleBehavior - from kivymd.uix.boxlayout import MDBoxLayout - - KV = ''' - MDFloatLayout: - - ElevatedWidget: - pos_hint: {'center_x': .5, 'center_y': .5} - size_hint: None, None - size: 100, 100 - md_bg_color: 0, 0, 1, 1 - ''' - - - class ElevatedWidget( - ThemableBehavior, - FakeRectangularElevationBehavior, - RectangularRippleBehavior, - ButtonBehavior, - MDBoxLayout, - ): - shadow_animation = ObjectProperty() - - def on_press(self, *args): - if self.shadow_animation: - Animation.cancel_all(self, "_elevation") - self.shadow_animation = Animation(_elevation=self.elevation + 10, d=0.4) - self.shadow_animation.start(self) - - def on_release(self, *args): - if self.shadow_animation: - Animation.cancel_all(self, "_elevation") - self.shadow_animation = Animation(_elevation=self.elevation, d=0.1) - self.shadow_animation.start(self) - - - class Example(MDApp): - def build(self): - return Builder.load_string(KV) - - - Example().run() - -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/rectangular-elevation-animation-effect.gif - :align: center - -Lighting position ------------------ - -.. code-block:: python - - from kivy.lang import Builder - - from kivymd.app import MDApp - from kivymd.uix.card import MDCard - from kivymd.uix.boxlayout import MDBoxLayout - from kivymd.uix.behaviors import RectangularElevationBehavior - - KV = ''' - MDScreen: - - ShadowCard: - pos_hint: {'center_x': .5, 'center_y': .5} - size_hint: None, None - size: 100, 100 - shadow_pos: -10 + slider.value, -10 + slider.value - elevation: 24 - md_bg_color: 1, 1, 1, 1 - - MDSlider: - id: slider - max: 20 - size_hint_x: .6 - pos_hint: {'center_x': .5, 'center_y': .3} - ''' - - - class ShadowCard(RectangularElevationBehavior, MDBoxLayout): - pass - - - class Example(MDApp): - def build(self): - return Builder.load_string(KV) - - - Example().run() - -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-pos.gif - :align: center -""" - -__all__ = ( - "CommonElevationBehavior", - "RectangularElevationBehavior", - "CircularElevationBehavior", - "RoundedRectangularElevationBehavior", - "ObservableShadow", - "FakeRectangularElevationBehavior", - "FakeCircularElevationBehavior", -) - -from io import BytesIO -from weakref import WeakMethod, ref - -from kivy import Logger -from kivy.clock import Clock -from kivy.core.image import Image as CoreImage -from kivy.lang import Builder -from kivy.metrics import dp -from kivy.properties import ( - AliasProperty, - BooleanProperty, - BoundedNumericProperty, - ListProperty, - NumericProperty, - ObjectProperty, - ReferenceListProperty, - StringProperty, - VariableListProperty, -) -from kivy.uix.widget import Widget -from PIL import Image, ImageDraw, ImageFilter - -from kivymd.app import MDApp - -Builder.load_string( - """ -#:import InstructionGroup kivy.graphics.instructions.InstructionGroup - - - - canvas.before: - # SOFT SHADOW - PushMatrix - Rotate: - angle: self.angle - origin: self._shadow_origin - Color: - group: "soft_shadow" - rgba: root.soft_shadow_cl - Rectangle: - group: "soft_shadow" - texture: self._soft_shadow_texture - size: self.soft_shadow_size - pos: self.soft_shadow_pos - PopMatrix - - # HARD SHADOW - PushMatrix - Rotate: - angle: self.angle - origin: self.center - Color: - group: "hard_shadow" - rgba: root.hard_shadow_cl - Rectangle: - group: "hard_shadow" - texture: self.hard_shadow_texture - size: self.hard_shadow_size - pos: self.hard_shadow_pos - PopMatrix - Color: - group: "shadow" - a: 1 -""", - filename="CommonElevationBehavior.kv", -) - - -class CommonElevationBehavior(Widget): - """Common base class for rectangular and circular elevation behavior.""" - - elevation = BoundedNumericProperty(0, min=0, errorvalue=0) - """ - Elevation of the widget. - - .. note:: - Although, this value does not represent the current elevation of the - widget. :attr:`~CommonElevationBehavior._elevation` can be used to - animate the current elevation and come back using the - :attr:`~CommonElevationBehavior.elevation` property directly. - - For example: + .. tab:: Declarative style with KV .. code-block:: python @@ -431,52 +24,41 @@ class CommonElevationBehavior(Widget): from kivy.uix.behaviors import ButtonBehavior from kivymd.app import MDApp - from kivymd.uix.behaviors import CircularElevationBehavior, CircularRippleBehavior - from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.behaviors import ( + RectangularRippleBehavior, + BackgroundColorBehavior, + CommonElevationBehavior, + ) KV = ''' - #:import Animation kivy.animation.Animation + + size_hint: None, None + size: "250dp", "50dp" - - size_hint: [None, None] - elevation: 6 - animation_: None - md_bg_color: [1] * 4 - on_size: - self.radius = [self.height / 2] * 4 - on_press: - if self.animation_: \ - self.animation_.cancel(self); \ - self.animation_ = Animation(_elevation=self.elevation + 6, d=0.08); \ - self.animation_.start(self) - on_release: - if self.animation_: \ - self.animation_.cancel(self); \ - self.animation_ = Animation(_elevation = self.elevation, d=0.08); \ - self.animation_.start(self) + MDScreen: - MDFloatLayout: + # With elevation effect + RectangularElevationButton: + pos_hint: {"center_x": .5, "center_y": .6} + elevation: 4.5 + shadow_offset: 0, 6 - WidgetWithShadow: - size: [root.size[1] / 2] * 2 - pos_hint: {"center": [0.5, 0.5]} + # Without elevation effect + RectangularElevationButton: + pos_hint: {"center_x": .5, "center_y": .4} ''' - class WidgetWithShadow( - CircularElevationBehavior, - CircularRippleBehavior, + class RectangularElevationButton( + RectangularRippleBehavior, + CommonElevationBehavior, ButtonBehavior, - MDBoxLayout, + BackgroundColorBehavior, ): def __init__(self, **kwargs): - # always set the elevation before the super().__init__ call - # self.elevation = 6 super().__init__(**kwargs) - - def on_size(self, *args): - self.radius = [self.size[0] / 2] + self.md_bg_color = "red" class Example(MDApp): @@ -486,990 +68,831 @@ class CommonElevationBehavior(Widget): Example().run() + .. tab:: Declarative python style + + .. code-block:: python + + from kivy.uix.behaviors import ButtonBehavior + + from kivymd.app import MDApp + from kivymd.uix.behaviors import ( + RectangularRippleBehavior, + BackgroundColorBehavior, + CommonElevationBehavior, + ) + from kivymd.uix.screen import MDScreen + + + 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") + + + 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}, + ), + ) + ) + + + Example().run() + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/rectangular-elevation-effect.png + :align: center + +.. warning:: + + 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`. + +Similarly, create a circular button: + +.. tabs:: + + .. tab:: Declarative style with KV + + .. code-block:: python + + from kivy.lang import Builder + from kivy.uix.behaviors import ButtonBehavior + + from kivymd.app import MDApp + from kivymd.uix.behaviors import CircularRippleBehavior, CommonElevationBehavior + from kivymd.uix.floatlayout import MDFloatLayout + + KV = ''' + + 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 + ''' + + + class CircularElevationButton( + CommonElevationBehavior, + CircularRippleBehavior, + ButtonBehavior, + MDFloatLayout, + ): + pass + + + class Example(MDApp): + def build(self): + return Builder.load_string(KV) + + + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivy.metrics import dp + from kivy.uix.behaviors import ButtonBehavior + + 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 + + + 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, + ) + ) + + + class Example(MDApp): + def build(self): + return ( + MDScreen( + CircularElevationButton( + pos_hint={"center_x": .5, "center_y": .5}, + elevation=4, + ) + ) + ) + + + Example().run() + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/circular-elevation-effect.png + :align: center + +Animating the elevation +----------------------- + +.. tabs:: + + .. tab:: Declarative style with KV + + .. code-block:: python + + from kivy.animation import Animation + from kivy.lang import Builder + from kivy.uix.behaviors import ButtonBehavior + + from kivymd.app import MDApp + from kivymd.uix.behaviors import CommonElevationBehavior, RectangularRippleBehavior + from kivymd.uix.widget import MDWidget + + 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 + ''' + + + class ElevatedWidget( + CommonElevationBehavior, + RectangularRippleBehavior, + ButtonBehavior, + MDWidget, + ): + _elev = 0 # previous elevation value + + def on_press(self, *args): + if not self._elev: + self._elev = self.elevation + Animation(elevation=self.elevation + 2, d=0.4).start(self) + + def on_release(self, *args): + Animation.cancel_all(self, "elevation") + Animation(elevation=self._elev, d=0.1).start(self) + + + class Example(MDApp): + def build(self): + return Builder.load_string(KV) + + + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivy.animation import Animation + from kivy.uix.behaviors import ButtonBehavior + + from kivymd.app import MDApp + from kivymd.uix.behaviors import CommonElevationBehavior, RectangularRippleBehavior + from kivymd.uix.screen import MDScreen + from kivymd.uix.widget import MDWidget + + + class ElevatedWidget( + CommonElevationBehavior, + RectangularRippleBehavior, + ButtonBehavior, + MDWidget, + ): + _elev = 0 # previous elevation value + + def on_press(self, *args): + if not self._elev: + self._elev = self.elevation + Animation(elevation=self.elevation + 2, d=0.4).start(self) + + def on_release(self, *args): + Animation.cancel_all(self, "elevation") + Animation(elevation=self._elev, d=0.1).start(self) + + + 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, + ) + ) + ) + + + Example().run() + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/rectangular-elevation-animation-effect.gif + :align: center +""" + +from __future__ import annotations + +__all__ = ( + "CommonElevationBehavior", + "RectangularElevationBehavior", + "CircularElevationBehavior", + "RoundedRectangularElevationBehavior", + "FakeRectangularElevationBehavior", + "FakeCircularElevationBehavior", +) + +import os + +from kivy import Logger +from kivy.clock import Clock +from kivy.core.window import Window +from kivy.graphics import RenderContext, RoundedRectangle +from kivy.properties import ( + AliasProperty, + BooleanProperty, + BoundedNumericProperty, + ColorProperty, + ListProperty, + NumericProperty, + ObjectProperty, + VariableListProperty, +) +from kivy.uix.widget import Widget + +from kivymd import glsl_path + + +# FIXME: Add shadow manipulation with canvas instructions such as +# PushMatrix and PopMatrix. +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`. """ - # Shadow rendering properties. - # Shadow rotation memory - SHARED ACROSS OTHER CLASSES. - angle = NumericProperty(0) - """ - Angle of rotation in degrees of the current shadow. - This value is shared across different widgets. - - .. note:: - This value will affect both, hard and soft shadows. - Each shadow has his own origin point that's computed every time the - elevation changes. - - .. warning:: - Do not add `PushMatrix` inside the canvas before and add `PopMatrix` - in the next layer, this will cause visual errors, because the stack - used will clip the push and pop matrix already inside the canvas.before - canvas layer. - - Incorrect: - - .. code-block:: kv - - - canvas.before: - PushMatrix - [...] - canvas: - PopMatrix - - Correct: - - .. code-block:: kv - - - canvas.before: - PushMatrix - [...] - PopMatrix - - - - :attr:`angle` is an :class:`~kivy.properties.NumericProperty` - and defaults to `0`. - """ - - radius = VariableListProperty([0]) + shadow_radius = VariableListProperty([0], length=4) """ Radius of the corners of the shadow. - This values represents each corner of the shadow, starting from `top-left` - corner and going clockwise. + + .. versionadded:: 1.1.0 + + 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: .. code-block:: python - radius = [ - "top-left", - "top-right", - "bottom-right", - "bottom-left", - ] + from kivy.lang import Builder - This value can be expanded thus allowing this settings to be valid: + from kivymd.app import MDApp - .. code-block:: python + KV = ''' + MDScreen: - widget.radius=[0] # Translates to [0, 0, 0, 0] - widget.radius=[10, 3] # Translates to [10, 3, 10, 3] - widget.radius=[7.0, 8.7, 1.5, 3.0] # Translates to [7, 8, 1, 3] + 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) + ''' + + + class Test(MDApp): + def build(self): + return Builder.load_string(KV) + + + Test().run() + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-radius.png + :align: center .. note:: - This value will affect both, hard and soft shadows. - This value only affects :class:`~RoundedRectangularElevationBehavior` - for now, but can be stored and used by custom shadow draw functions. + 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. - :attr:`radius` is an :class:`~kivy.properties.VariableListProperty` + .. code-block:: python + + shadow_radius = ['top-right', 'bot-right', 'top-left', 'bot-left'] + kivy_radius = ['top-left', 'top-right', 'bottom-right', 'bottom-left'] + + :attr:`shadow_radius` is an :class:`~kivy.properties.VariableListProperty` and defaults to `[0, 0, 0, 0]`. """ - # Position of the shadow. - _shadow_origin_x = NumericProperty(0) + shadow_softness = NumericProperty(12) """ - Shadow origin `x` position for the rotation origin. + Softness of the shadow. - Managed by `_shadow_origin`. - - :attr:`_shadow_origin_x` is an :class:`~kivy.properties.NumericProperty` - and defaults to `0`. - - .. note:: - This property is automatically processed. by _shadow_origin. - """ - - _shadow_origin_y = NumericProperty(0) - """ - Shadow origin y position for the rotation origin. - - Managed by :attr:`_shadow_origin`. - - :attr:`_shadow_origin_y` is an :class:`~kivy.properties.NumericProperty` - and defaults to `0`. - - .. note:: - This property is automatically processed. - """ - - _shadow_origin = ReferenceListProperty(_shadow_origin_x, _shadow_origin_y) - """ - Soft shadow rotation origin point. - - :attr:`_shadow_origin` is an :class:`~kivy.properties.ReferenceListProperty` - and defaults to `[0, 0]`. - - .. note:: - This property is automatically processed and relative to the canvas center. - """ - - _shadow_pos = ListProperty([0, 0]) # custom offset - """ - Soft shadow origin point. - - :attr:`_shadow_pos` is an :class:`~kivy.properties.ListProperty` - and defaults to `[0, 0]`. - - .. note:: - This property is automatically processed and relative to the widget's - canvas center. - """ - - shadow_pos = ListProperty([0, 0]) # bottom left corner - """ - Custom shadow origin point. If this property is set, :attr:`_shadow_pos` - will be ommited. - - This property allows users to fake light source. - - :attr:`shadow_pos` is an :class:`~kivy.properties.ListProperty` - and defaults to `[0, 0]`. - - .. note:: - this value overwrite the :attr:`_shadow_pos` processing. - """ - - # Shadow Group shared memory - __shadow_groups = {"global": []} - - shadow_group = StringProperty("global") - """ - Widget's shadow group. - By default every widget with a shadow is saved inside the memory - :attr:`__shadow_groups` as a weakref. This means that you can have multiple - light sources, one for every shadow group. - - To fake a light source use :attr:`force_shadow_pos`. - - :attr:`shadow_group` is an :class:`~kivy.properties.StringProperty` - and defaults to `"global"`. - """ - - _elevation = BoundedNumericProperty(0, min=0, errorvalue=0) - """ - Current elevation of the widget. - - .. warning:: - This property is the current elevation of the widget, do not - use this property directly, instead, use :class:`~CommonElevationBehavior` - elevation. - - :attr:`_elevation` is an :class:`~kivy.properties.NumericProperty` - and defaults to `0`. - """ - - # soft shadow - _soft_shadow_texture = ObjectProperty() - """ - Texture of the soft shadow texture for the canvas. - - :attr:`_soft_shadow_texture` is an :class:`~kivy.core.image.Image` - and defaults to `None`. - - .. note:: - This property is automatically processed. - """ - - soft_shadow_size = ListProperty([0, 0]) - """ - Size of the soft shadow texture over the canvas. - - :attr:`soft_shadow_size` is an :class:`~kivy.properties.ListProperty` - and defaults to `[0, 0]`. - - .. note:: - This property is automatically processed. - """ - - soft_shadow_pos = ListProperty([0, 0]) - """ - Position of the hard shadow texture over the canvas. - - :attr:`soft_shadow_pos` is an :class:`~kivy.properties.ListProperty` - and defaults to `[0, 0]`. - - .. note:: - This property is automatically processed. - """ - - soft_shadow_cl = ListProperty([0, 0, 0, 0.50]) - """ - Color of the soft shadow. - - :attr:`soft_shadow_cl` is an :class:`~kivy.properties.ListProperty` - and defaults to `[0, 0, 0, 0.15]`. - """ - - # hard shadow - hard_shadow_texture = ObjectProperty() - """ - Texture of the hard shadow texture for the canvas. - - :attr:`hard_shadow_texture` is an :class:`~kivy.core.image.Image` - and defaults to `None`. - - .. note:: - This property is automatically processed when elevation is changed. - """ - - hard_shadow_size = ListProperty([0, 0]) - """ - Size of the hard shadow texture over the canvas. - - :attr:`hard_shadow_size` is an :class:`~kivy.properties.ListProperty` - and defaults to `[0, 0]`. - - .. note:: - This property is automatically processed when elevation is changed. - """ - - hard_shadow_pos = ListProperty([0, 0]) - """ - Position of the hard shadow texture over the canvas. - - :attr:`hard_shadow_pos` is an :class:`~kivy.properties.ListProperty` - and defaults to `[0, 0]`. - - .. note:: - This property is automatically processed when elevation is changed. - """ - - hard_shadow_cl = ListProperty([0, 0, 0, 0.15]) - """ - Color of the hard shadow. - - .. note:: - :attr:`hard_shadow_cl` is an :class:`~kivy.properties.ListProperty` - and defaults to `[0, 0, 0, 0.15]`. - """ - - # Shared property for some calculations. - # This values are used to improve the gaussain blur and avoid that - # the blur goes outside the texture. - hard_shadow_offset = BoundedNumericProperty( - 2, min=0, errorhandler=lambda x: 0 if x < 0 else x - ) - """ - This value sets a special offset to the shadow canvas, this offset allows a - correct draw of the canvas size. allowing the effect to correctly blur the - image in the given space. - - :attr:`hard_shadow_offset` is an :class:`~kivy.properties.BoundedNumericProperty` - and defaults to `2`. - """ - - soft_shadow_offset = BoundedNumericProperty( - 4, min=0, errorhandler=lambda x: 0 if x < 0 else x - ) - """ - This value sets a special offset to the shadow canvas, this offset allows a - correct draw of the canvas size. allowing the effect to correctly blur the - image in the given space. - - :attr:`soft_shadow_offset` is an :class:`~kivy.properties.BoundedNumericProperty` - and defaults to `4`. - """ - - draw_shadow = ObjectProperty(None) - """ - This property controls the draw call of the context. - - This property is automatically set to :attr:`__draw_shadow__` inside the - `super().__init__ call.` unless the property is different of None. - - To set a different drawing instruction function, set this property before the - `super(),__init__` call inside the `__init__` definition of the new class. - - You can use the source for this classes as example of how to draw over - with the context: - - Real time shadows: - #. :class:`~RectangularElevationBehavior` - #. :class:`~CircularElevationBehavior` - #. :class:`~RoundedRectangularElevationBehavior` - #. :class:`~ObservableShadow` - - - Fake shadows (d`ont use this property): - #. :class:`~FakeRectangularElevationBehavior` - #. :class:`~FakeCircularElevationBehavior` - - :attr:`draw_shadow` is an :class:`~kivy.properties.ObjectProperty` - and defaults to `None`. - - .. note:: If this property is left to `None` the - :class:`~CommonElevationBehavior` will set to a function that will - raise a `NotImplementedError` inside `super().__init__`. - - Follow the next example to set a new draw instruction for the class - inside `__init__`: + .. versionadded:: 1.1.0 .. code-block:: python - class RoundedRectangularElevationBehavior(CommonElevationBehavior): - ''' - Shadow class for the RoundedRectangular shadow behavior. - Controls the size and position of the shadow. - ''' + from kivy.lang import Builder - def __init__(self, **kwargs): - self._draw_shadow = WeakMethod(self.__draw_shadow__) - super().__init__(**kwargs) + from kivymd.app import MDApp + from kivymd.uix.behaviors import BackgroundColorBehavior, CommonElevationBehavior - def __draw_shadow__(self, origin, end, context=None): - context.draw(...) + KV = ''' + + size_hint: None, None + size: "250dp", "50dp" - Context is a `Pillow` `ImageDraw` class. For more information check the - [Pillow official documentation](https://github.com/python-pillow/Pillow/). + + MDScreen: + + RectangularElevationButton: + pos_hint: {"center_x": .5, "center_y": .6} + elevation: 6 + shadow_softness: 6 + + RectangularElevationButton: + pos_hint: {"center_x": .5, "center_y": .4} + elevation: 6 + shadow_softness: 12 + ''' + + + class RectangularElevationButton(CommonElevationBehavior, BackgroundColorBehavior): + md_bg_color = [0, 0, 1, 1] + + + class Example(MDApp): + def build(self): + return Builder.load_string(KV) + + + Example().run() + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-softness.png + :align: center + + :attr:`shadow_softness` is an :class:`~kivy.properties.NumericProperty` + and defaults to `12`. """ - # All classes that uses a fake shadow shall set this value as `True` - # for performance. - _fake_elevation = BooleanProperty(False) + shadow_offset = ListProperty((0, 2)) + """ + Offset of the shadow. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + from kivy.lang import Builder + + from kivymd.app import MDApp + from kivymd.uix.behaviors import BackgroundColorBehavior, CommonElevationBehavior + + KV = ''' + + size_hint: None, None + size: "100dp", "100dp" + + + MDScreen: + + RectangularElevationButton: + pos_hint: {"center_x": .5, "center_y": .5} + elevation: 6 + shadow_radius: 18 + shadow_softness: 24 + shadow_offset: 12, 12 + ''' + + + class RectangularElevationButton(CommonElevationBehavior, BackgroundColorBehavior): + md_bg_color = [0, 0, 1, 1] + + + class Example(MDApp): + def build(self): + return Builder.load_string(KV) + + + Example().run() + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-offset-1.png + :align: center + + .. code-block:: kv + + RectangularElevationButton: + shadow_offset: -12, 12 + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-offset-2.png + :align: center + + .. code-block:: kv + + RectangularElevationButton: + shadow_offset: -12, -12 + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-offset-3.png + :align: center + + .. code-block:: kv + + RectangularElevationButton: + shadow_offset: 12, -12 + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-offset-4.png + :align: center + + :attr:`shadow_offset` is an :class:`~kivy.properties.ListProperty` + and defaults to `(0, 2)`. + """ + + shadow_color = ColorProperty([0, 0, 0, 0.6]) + """ + Offset of the shadow. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + RectangularElevationButton: + shadow_color: 0, 0, 1, .8 + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadow-color.png + :align: center + + :attr:`shadow_color` is an :class:`~kivy.properties.ColorProperty` + and defaults to `[0.4, 0.4, 0.4, 0.8]`. + """ + + _transition_ref = ObjectProperty() + _has_relative_position = BooleanProperty(defaultvalue=False) + _elevation = 0 + _shadow_color = [0.0, 0.0, 0.0, 0.0] + + 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]] + + def _set_window_pos(self, value): + self.window_pos = value + + window_pos = AliasProperty(_get_window_pos, _set_window_pos) def __init__(self, **kwargs): - if self.draw_shadow is None: - self.draw_shadow = WeakMethod(self.__draw_shadow__) - self.prev_shadow_group = None - im = BytesIO() - Image.new("RGBA", (4, 4), color=(0, 0, 0, 0)).save(im, format="png") - im.seek(0) - # Setting a empty image as texture, improves performance. - self._soft_shadow_texture = self.hard_shadow_texture = CoreImage( - im, ext="png" - ).texture - Clock.schedule_once(self.shadow_preset, -1) - self.on_shadow_group(self, self.shadow_group) - - self.bind( - pos=self._update_shadow, - size=self._update_shadow, - radius=self._update_shadow, - ) super().__init__(**kwargs) - def on_shadow_group(self, instance, value): + with self.canvas.before: + self.context = RenderContext(use_parent_projection=True) + with self.context: + self.rect = RoundedRectangle(pos=self.pos, size=self.size) + + self.after_init() + + 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: """ - This function controls the shadow group of the widget. - Do not use Directly to change the group. instead, use the shadow_group - :attr:`property`. + Checks if the widget has relative properties and if necessary + binds Window.on_draw and screen events to fix behavior """ - groups = CommonElevationBehavior.__shadow_groups - if self.prev_shadow_group: - group = groups[self.prev_shadow_group] - for widget in group[:]: - if widget() is self: - group.remove(widget) - group = self.prev_shadow_group = self.shadow_group - if group not in groups: - groups[group] = [] - r = ref(self, CommonElevationBehavior._clear_shadow_groups) - groups[group].append(r) + if self.pos != self.window_pos: + self._has_relative_position = True - @staticmethod - def _clear_shadow_groups(wk): - # auto flush the element when the weak reference have been deleted - groups = CommonElevationBehavior.__shadow_groups - for group in list(groups.values()): - if not group: - break - if wk in group: - group.remove(wk) + # 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 - def force_shadow_pos(self, shadow_pos): - """ - This property forces the shadow position in every widget inside the - widget. The argument :attr:`shadow_pos` is expected as a - or . - """ - - if self.shadow_group is None: - return - group = CommonElevationBehavior.__shadow_groups[self.shadow_group] - for wk in group[:]: - widget = wk() - if widget is None: - group.remove(wk) - widget.shadow_pos = shadow_pos - del group - - def update_group_property(self, property_name, value): - """ - This functions allows to change properties of every widget inside the - shadow group. - """ - - if self.shadow_group is None: - return - group = CommonElevationBehavior.__shadow_groups[self.shadow_group] - for wk in group[:]: - widget = wk() - if widget is None: - group.remove(wk) - setattr(widget, property_name, value) - del group - - def shadow_preset(self, *args): - """ - This function is meant to set the default configuration of the - elevation. - - After a new instance is created, the elevation property will be launched - and thus this function will update the elevation if the KV lang have not - done it already. - - Works similar to an `__after_init__` call inside a widget. - """ - - from kivymd.uix.card import MDCard - - if self.elevation is None and not issubclass(self.__class__, MDCard): - self.elevation = 10 - if self._fake_elevation is False: - self._update_shadow(self, self.elevation) - self.bind( - pos=self._update_shadow, - size=self._update_shadow, - _elevation=self._update_shadow, - ) - - def on_elevation(self, instance, value): - """ - Elevation event that sets the current elevation value to `_elevation`. - """ - - if value is not None: - self._elevation = value - - def _set_soft_shadow_a(self, value): - value = 0 if value < 0 else (1 if value > 1 else value) - self.soft_shadow_cl[-1] = value - return True - - def _set_hard_shadow_a(self, value): - value = 0 if value < 0 else (1 if value > 1 else value) - self.hard_shadow_cl[-1] = value - return True - - def _get_soft_shadow_a(self): - return self.soft_shadow_cl[-1] - - def _get_hard_shadow_a(self): - return self.hard_shadow_cl[-1] - - _soft_shadow_a = AliasProperty( - _get_soft_shadow_a, _set_soft_shadow_a, bind=["soft_shadow_cl"] - ) - _hard_shadow_a = AliasProperty( - _get_hard_shadow_a, _set_hard_shadow_a, bind=["hard_shadow_cl"] - ) - - def on_disabled(self, instance, value): - """ - This function hides the shadow when the widget is disabled. - It sets the shadow to `0`. - """ - - if self.disabled is True: - self._elevation = 0 - else: - self._elevation = 0 if self.elevation is None else self.elevation - self._update_shadow(self, self._elevation) - try: - super().on_disabled(instance, value) - except Exception: - pass - - def _update_elevation(self, instance, value): - self._elevation = value - self._update_shadow(instance, value) - - def _update_shadow_pos(self, instance, value): - if self._elevation > 0: - self.hard_shadow_pos = [ - self.x - dp(self.hard_shadow_offset), # + self.shadow_pos[0], - self.y - dp(self.hard_shadow_offset), # + self.shadow_pos[1], - ] - if self.shadow_pos == [0, 0]: - self.soft_shadow_pos = [ - self.x - + self._shadow_pos[0] - - self._elevation - - dp(self.soft_shadow_offset), - self.y - + self._shadow_pos[1] - - self._elevation - - dp(self.soft_shadow_offset), - ] + elif widget.parent and str(widget) != str(widget.parent): + widget = widget.parent else: - self.soft_shadow_pos = [ - self.x - + self.shadow_pos[0] - - self._elevation - - dp(self.soft_shadow_offset), - self.y - + self.shadow_pos[1] - - self._elevation - - dp(self.soft_shadow_offset), - ] - self._shadow_origin = [ - self.soft_shadow_pos[0] + self.soft_shadow_size[0] / 2, - self.soft_shadow_pos[1] + self.soft_shadow_size[1] / 2, + break + + 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 - def on__shadow_pos(self, ins, val): - """ - Updates the shadow with the computed value. + Clock.schedule_once(on_shadow_color) - Call this function every time you need to force a shadow update. - """ + 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)) - self._update_shadow_pos(ins, val) + Clock.schedule_once(on_shadow_radius) - def on_shadow_pos(self, ins, val): - """ - Updates the shadow with the fixed value. + def on_shadow_softness(self, instance, value) -> None: + def on_shadow_softness(*args): + if hasattr(self, "context"): + self.context["shadow_softness"] = float(value) - Call this function every time you need to force a shadow update. - """ + Clock.schedule_once(on_shadow_softness) - self._update_shadow_pos(ins, val) - - def _update_shadow(self, instance, value): - self._update_shadow_pos(instance, value) - if self._elevation > 0 and self._fake_elevation is False: - # dynamic elevation position for the shadow - if self.shadow_pos == [0, 0]: - self._shadow_pos = [0, -self._elevation * 0.4] - - # HARD Shadow - offset = int(dp(self.hard_shadow_offset)) - size = [ - int(self.size[0] + (offset * 2)), - int(self.size[1] + (offset * 2)), - ] - im = BytesIO() - # context - img = Image.new("RGBA", tuple(size), color=(0, 0, 0, 0)) - # draw context - shadow = ImageDraw.Draw(img) - self.draw_shadow()( - [offset, offset], - [ - int(size[0] - 1 - offset), - int(size[1] - 1 - offset), - ], - context=shadow - # context=ref(shadow) - ) - img = img.filter( - ImageFilter.GaussianBlur( - radius=int(dp(1 + self.hard_shadow_offset / 3)) + 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 ) - ) - img.save(im, format="png") - im.seek(0) - self.hard_shadow_size = size - self.hard_shadow_texture = CoreImage(im, ext="png").texture - # soft shadow - if self.soft_shadow_cl[-1] > 0: - offset = dp(self.soft_shadow_offset) - size = [ - int(self.size[0] + dp(self._elevation * 2) + (offset * 2)), - int(self.size[1] + dp(self._elevation * 2) + (offset * 2)), - # ((self._elevation)*2) + x + (offset*2)) for x in self.size - ] - im = BytesIO() - img = Image.new("RGBA", tuple(size), color=((0,) * 4)) - shadow = ImageDraw.Draw(img) - _offset = int(dp(self._elevation + offset)) - self.draw_shadow()( - [ - _offset, - _offset, - ], - [int(size[0] - _offset - 1), int(size[1] - _offset - 1)], - context=shadow - # context=ref(shadow) - ) - img = img.filter( - ImageFilter.GaussianBlur(radius=self._elevation // 2) - ) - shadow = ImageDraw.Draw(img) - img.save(im, format="png") - im.seek(0) - self.soft_shadow_size = size - self._soft_shadow_texture = CoreImage(im, ext="png").texture - else: - im = BytesIO() - Image.new("RGBA", (4, 4), color=(0, 0, 0, 0)).save(im, format="png") - im.seek(0) - self._soft_shadow_texture = self.hard_shadow_texture = CoreImage( - im, ext="png" - ).texture + Clock.schedule_once(on_elevation) + + def on_shadow_offset(self, instance, value) -> None: + self.on_size() + self.on_pos() + + def update_window_position(self, *args) -> None: + """ + This function is used only when the widget has relative position + properties. + """ + + self.on_pos() + + def on_pos(self, *args) -> None: + if not hasattr(self, "rect"): return - def _get_center(self): - center = [self.pos[0] + self.width / 2, self.pos[1] + self.height / 2] - return center + if ( + self._has_relative_position + and not self.context.use_parent_modelview + ): + pos = self.window_pos + else: + pos = self.pos - def __draw_shadow__(self, origin, end, context=None): - Logger.warning( - f"KivyMD: " - f"If you see this error, this means that either youre using " - f"`CommonElevationBehavio`r directly or your 'shader' dont have a " - f"`_draw_shadow` instruction, remember to overwrite this function" - f"to draw over the image context. Тhe figure you would like. " - f"Or your class {self.__class__.__name__} is not inherited from " - f"any of the classes {__all__}" + 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], + ] + + 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() + + def on_size(self, *args) -> None: + if not hasattr(self, "rect"): + return + + # 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), ) + 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() + + def on_opacity(self, instance, value: int | float) -> None: + """ + Adjusts the transparency of the shadow according to the transparency + of the widget. + """ + + def on_opacity(*args): + self._shadow_color = list(map(float, self._shadow_color))[:-1] + [ + float(value) + ] + self.context["shadow_color"] = self._shadow_color + + super().on_opacity(instance, value) + Clock.schedule_once(on_opacity) + + def on_radius(self, instance, value) -> None: + self.shadow_radius = [value[1], value[2], value[0], value[3]] + + def on_disabled(self, instance, value) -> None: + if value: + self._elevation = 0 + self.hide_elevation(True) + else: + self.hide_elevation(False) + + 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)] + + self.on_shadow_color(self, self._shadow_color) + self.on_size() + self.on_pos() + class RectangularElevationBehavior(CommonElevationBehavior): """ - Base class for a rectangular elevation behavior. + .. deprecated:: 1.1.0 + Use :class:`~CommonElevationBehavior` class instead. """ def __init__(self, **kwargs): - self.draw_shadow = WeakMethod(self.__draw_shadow__) super().__init__(**kwargs) - - def __draw_shadow__(self, origin, end, context=None): - context.rectangle(origin + end, fill=tuple([255] * 4)) + Logger.warning( + "KivyMD: " + "The `RectangularElevationBehavior` class has been deprecated. " + "Use the `CommonElevationBehavior` class instead.`" + ) class CircularElevationBehavior(CommonElevationBehavior): """ - Base class for a circular elevation behavior. + .. deprecated:: 1.1.0 + Use :class:`~CommonElevationBehavior` class instead. """ def __init__(self, **kwargs): - self.draw_shadow = WeakMethod(self.__draw_shadow__) super().__init__(**kwargs) - - def __draw_shadow__(self, origin, end, context=None): - context.ellipse(origin + end, fill=tuple([255] * 4)) + Logger.warning( + "KivyMD: " + "The `CircularElevationBehavior` class has been deprecated. " + "Use the `CommonElevationBehavior` class instead.`" + ) class RoundedRectangularElevationBehavior(CommonElevationBehavior): """ - Base class for rounded rectangular elevation behavior. + .. deprecated:: 1.1.0 + Use :class:`~CommonElevationBehavior` class instead. """ def __init__(self, **kwargs): - self.bind( - radius=self._update_shadow, - ) - self.draw_shadow = WeakMethod(self.__draw_shadow__) super().__init__(**kwargs) - - def __draw_shadow__(self, origin, end, context=None): - if self.radius == [0, 0, 0, 0]: - context.rectangle(origin + end, fill=tuple([255] * 4)) - else: - radius = [x * 2 for x in self.radius] - context.pieslice( - [ - origin[0], - origin[1], - origin[0] + radius[0], - origin[1] + radius[0], - ], - 180, - 270, - fill=(255, 255, 255, 255), - ) - context.pieslice( - [ - end[0] - radius[1], - origin[1], - end[0], - origin[1] + radius[1], - ], - 270, - 360, - fill=(255, 255, 255, 255), - ) - context.pieslice( - [ - end[0] - radius[2], - end[1] - radius[2], - end[0], - end[1], - ], - 0, - 90, - fill=(255, 255, 255, 255), - ) - context.pieslice( - [ - origin[0], - end[1] - radius[3], - origin[0] + radius[3], - end[1], - ], - 90, - 180, - fill=(255, 255, 255, 255), - ) - if all((x == self.radius[0] for x in self.radius)): - radius = int(self.radius[0]) - context.rectangle( - [ - origin[0] + radius, - origin[1], - end[0] - radius, - end[1], - ], - fill=(255,) * 4, - ) - context.rectangle( - [ - origin[0], - origin[1] + radius, - end[0], - end[1] - radius, - ], - fill=(255,) * 4, - ) - else: - radius = [ - max((self.radius[0], self.radius[1])), - max((self.radius[1], self.radius[2])), - max((self.radius[2], self.radius[3])), - max((self.radius[3], self.radius[0])), - ] - context.rectangle( - [ - origin[0] + self.radius[0], - origin[1], - end[0] - self.radius[1], - end[1] - radius[2], - ], - fill=(255,) * 4, - ) - context.rectangle( - [ - origin[0] + radius[3], - origin[1] + self.radius[1], - end[0], - end[1] - self.radius[2], - ], - fill=(255,) * 4, - ) - context.rectangle( - [ - origin[0] + self.radius[3], - origin[1] + radius[0], - end[0] - self.radius[2], - end[1], - ], - fill=(255,) * 4, - ) - context.rectangle( - [ - origin[0], - origin[1] + self.radius[0], - end[0] - radius[2], - end[1] - self.radius[3], - ], - fill=(255,) * 4, - ) - - -class ObservableShadow(CommonElevationBehavior): - """ - ObservableShadow is real time shadow render that it's intended to only - render a partial shadow of widgets based upon on the window observable - area, this is meant to improve the performance of bigger widgets. - - .. warning:: - This is an empty class, the name has been reserved for future use. - if you include this clas in your object, you wil get a - `NotImplementedError`. - """ - - def __init__(self, **kwargs): - # self._shadow = MDApp.get_running_app().theme_cls.round_shadow - # self._fake_elevation=True - raise NotImplementedError( - "ObservableShadow:\n\t" "This class is in current development" + Logger.warning( + "KivyMD: " + "The `RoundedRectangularElevationBehavior` class has been " + "deprecated. Use the `CommonElevationBehavior` class instead.`" ) - super().__init__(**kwargs) class FakeRectangularElevationBehavior(CommonElevationBehavior): """ - `FakeRectangularElevationBehavio`r is a shadow mockup for widgets. Improves - performance using cached images inside `kivymd.images` dir - - This class cast a fake Rectangular shadow behaind the widget. - - You can either use this behavior to overwrite the elevation of a prefab - widget, or use it directly inside a new widget class definition. - - Use this class as follows for new widgets: - - .. code-block:: python - - class NewWidget( - ThemableBehavior, - FakeCircularElevationBehavior, - SpecificBackgroundColorBehavior, - # here you add the other front end classes for the widget front_end, - ): - [...] - - With this method each class can draw it's content in the canvas in the - correct order, avoiding some visual errors. - - `FakeCircularElevationBehavior` will load prefabricated textures to - optimize loading times. - - .. note:: About rounded corners: - be careful, since this behavior is a mockup and will not draw any - rounded corners. + .. deprecated:: 1.1.0 + Use :class:`~CommonElevationBehavior` class instead. """ def __init__(self, **kwargs): - # self._shadow = MDApp.get_running_app().theme_cls.round_shadow - self.draw_shadow = WeakMethod(self.__draw_shadow__) - self._fake_elevation = True - self._update_shadow(self, self.elevation) super().__init__(**kwargs) - - def _update_shadow(self, *args): - if self._elevation > 0: - # Set shadow size. - ratio = self.width / (self.height if self.height != 0 else 1) - if -2 < ratio < 2: - self._shadow = MDApp.get_running_app().theme_cls.quad_shadow - width = soft_width = self.width * 1.9 - height = soft_height = self.height * 1.9 - elif ratio <= -2: - self._shadow = MDApp.get_running_app().theme_cls.rec_st_shadow - ratio = abs(ratio) - if ratio > 5: - ratio = ratio * 22 - else: - ratio = ratio * 11.5 - width = soft_width = self.width * 1.9 - height = self.height + dp(ratio) - soft_height = ( - self.height + dp(ratio) + dp(self._elevation) * 0.5 - ) - else: - self._shadow = MDApp.get_running_app().theme_cls.quad_shadow - width = soft_width = self.width * 1.8 - height = soft_height = self.height * 1.8 - - self.soft_shadow_size = (soft_width, soft_height) - self.hard_shadow_size = (width, height) - # Set ``soft_shadow`` parameters. - center_x, center_y = self._get_center() - self.hard_shadow_pos = self.soft_shadow_pos = ( - center_x - soft_width / 2, - center_y - soft_height / 2 - dp(self._elevation * 0.5), - ) - # Set transparency - self._soft_shadow_a = 0.1 * 1.05**self._elevation - self._hard_shadow_a = 0.4 * 0.8**self._elevation - t = int(round(self._elevation)) - if 0 < t <= 23: - self._soft_shadow_texture = ( - self._hard_shadow_texture - ) = self._shadow.textures[str(t)] - else: - self._soft_shadow_texture = ( - self._hard_shadow_texture - ) = self._shadow.textures["23"] - else: - self._soft_shadow_a = 0 - self._hard_shadow_a = 0 - - def __draw_shadow__(self, origin, end, context=None): - pass + Logger.warning( + "KivyMD: " + "The `FakeRectangularElevationBehavior` class has been " + "deprecated. Use the `CommonElevationBehavior` class instead." + ) class FakeCircularElevationBehavior(CommonElevationBehavior): """ - `FakeCircularElevationBehavior` is a shadow mockup for widgets. Improves - performance using cached images inside `kivymd.images` dir - - This class cast a fake elliptic shadow behaind the widget. - - You can either use this behavior to overwrite the elevation of a prefab - widget, or use it directly inside a new widget class definition. - - Use this class as follows for new widgets: - - .. code-block:: python - - class NewWidget( - ThemableBehavior, - FakeCircularElevationBehavior, - SpecificBackgroundColorBehavior, - # here you add the other front end classes for the widget front_end, - ): - [...] - - With this method each class can draw it's content in the canvas in the - correct order, avoiding some visual errors. - - `FakeCircularElevationBehavior` will load prefabricated textures to optimize - loading times. - - .. note:: About rounded corners: - be careful, since this behavior is a mockup and will not draw any rounded - corners. only perfect ellipses. + .. deprecated:: 1.1.0 + Use :class:`~CommonElevationBehavior` class instead. """ def __init__(self, **kwargs): - self._shadow = MDApp.get_running_app().theme_cls.round_shadow - self.draw_shadow = WeakMethod(self.__draw_shadow__) - self._fake_elevation = True - self._update_shadow(self, self.elevation) super().__init__(**kwargs) - - def _update_shadow(self, *args): - if self._elevation > 0: - # set shadow size - width = self.width * 2 - height = self.height * 2 - center_x, center_y = self._get_center() - - x = center_x - width / 2 - self.soft_shadow_size = (width, height) - self.hard_shadow_size = (width, height) - # set ``soft_shadow`` parameters - y = center_y - height / 2 - dp(0.5 * self._elevation) - self.soft_shadow_pos = (x, y) - - # set ``hard_shadow`` parameters - y = center_y - height / 2 - dp(0.5 * self._elevation) - self.hard_shadow_pos = (x, y) - - # shadow transparency - self._soft_shadow_a = 0.1 * 1.05**self._elevation - self._hard_shadow_a = 0.4 * 0.8**self._elevation - t = int(round(self._elevation)) - if 0 < t <= 23: - if hasattr(self, "_shadow"): - self._soft_shadow_texture = ( - self._hard_shadow_texture - ) = self._shadow.textures[str(t)] - else: - self._soft_shadow_texture = ( - self._hard_shadow_texture - ) = self._shadow.textures["23"] - else: - self._soft_shadow_a = 0 - self._hard_shadow_a = 0 - - def __draw_shadow__(self, origin, end, context=None): - pass + Logger.warning( + "KivyMD: " + "The `FakeCircularElevationBehavior` class has been deprecated. " + "Use the `CommonElevationBehavior` class instead." + ) diff --git a/sbapp/kivymd/uix/behaviors/ripple_behavior.py b/sbapp/kivymd/uix/behaviors/ripple_behavior.py index f06e724..1271878 100755 --- a/sbapp/kivymd/uix/behaviors/ripple_behavior.py +++ b/sbapp/kivymd/uix/behaviors/ripple_behavior.py @@ -111,7 +111,7 @@ from kivy.properties import ( from kivy.uix.behaviors import ToggleButtonBehavior -class CommonRipple(object): +class CommonRipple: """Base class for ripple effect.""" ripple_rad_default = NumericProperty(1) diff --git a/sbapp/kivymd/uix/behaviors/rotate_behavior.py b/sbapp/kivymd/uix/behaviors/rotate_behavior.py new file mode 100644 index 0000000..c80b879 --- /dev/null +++ b/sbapp/kivymd/uix/behaviors/rotate_behavior.py @@ -0,0 +1,133 @@ +""" +Behaviors/Rotate +================ + +.. versionadded:: 1.1.0 + +Base class for controlling the rotate of the widget. + +.. note:: See `kivy.graphics.Rotate + `_ + for more information. + +Kivy +---- + +.. code-block:: python + + from kivy.animation import Animation + from kivy.lang import Builder + from kivy.app import App + from kivy.properties import NumericProperty + from kivy.uix.button import Button + + KV = ''' + Screen: + + RotateButton: + size_hint: .5, .5 + pos_hint: {"center_x": .5, "center_y": .5} + on_release: app.change_rotate(self) + + canvas.before: + PushMatrix + Rotate: + angle: self.rotate_value_angle + axis: 0, 0, 1 + origin: self.center + canvas.after: + PopMatrix + ''' + + + class RotateButton(Button): + rotate_value_angle = NumericProperty(0) + + + class Test(App): + def build(self): + return Builder.load_string(KV) + + def change_rotate(self, instance_button: Button) -> None: + Animation(rotate_value_angle=45, d=0.3).start(instance_button) + + + Test().run() + +KivyMD +------ + +.. code-block:: python + + from kivy.animation import Animation + from kivy.lang import Builder + from kivy.uix.behaviors import ButtonBehavior + + from kivymd.app import MDApp + from kivymd.uix.behaviors import RotateBehavior + from kivymd.uix.boxlayout import MDBoxLayout + + KV = ''' + MDScreen: + + RotateBox: + size_hint: .5, .5 + pos_hint: {"center_x": .5, "center_y": .5} + on_release: app.change_rotate(self) + md_bg_color: "red" + ''' + + + class RotateBox(ButtonBehavior, RotateBehavior, MDBoxLayout): + pass + + + class Test(MDApp): + def build(self): + return Builder.load_string(KV) + + def change_rotate(self, instance_button: RotateBox) -> None: + Animation(rotate_value_angle=45, d=0.3).start(instance_button) + + + Test().run() +""" + +__all__ = ("RotateBehavior",) + +from kivy.lang import Builder +from kivy.properties import ListProperty, NumericProperty + +Builder.load_string( + """ + + canvas.before: + PushMatrix + Rotate: + angle: self.rotate_value_angle + axis: tuple(self.rotate_value_axis) + origin: self.center + canvas.after: + PopMatrix +""" +) + + +class RotateBehavior: + """Base class for controlling the rotate of the widget.""" + + rotate_value_angle = NumericProperty(0) + """ + Property for getting/setting the angle of the rotation. + + :attr:`rotate_value_angle` is an :class:`~kivy.properties.NumericProperty` + and defaults to `0`. + """ + + rotate_value_axis = ListProperty((0, 0, 1)) + """ + Property for getting/setting the axis of the rotation. + + :attr:`rotate_value_axis` is an :class:`~kivy.properties.ListProperty` + and defaults to `(0, 0, 1)`. + """ diff --git a/sbapp/kivymd/uix/behaviors/scale_behavior.py b/sbapp/kivymd/uix/behaviors/scale_behavior.py new file mode 100644 index 0000000..bfc1e03 --- /dev/null +++ b/sbapp/kivymd/uix/behaviors/scale_behavior.py @@ -0,0 +1,156 @@ +""" +Behaviors/Scale +=============== + +.. versionadded:: 1.1.0 + +Base class for controlling the scale of the widget. + +.. note:: See `kivy.graphics.Rotate + `_ + for more information. + +Kivy +---- + +.. code-block:: python + + from kivy.animation import Animation + from kivy.lang import Builder + from kivy.properties import NumericProperty + from kivy.uix.button import Button + from kivy.app import App + + + KV = ''' + Screen: + + ScaleButton: + size_hint: .5, .5 + pos_hint: {"center_x": .5, "center_y": .5} + on_release: app.change_scale(self) + + canvas.before: + PushMatrix + Scale: + x: self.scale_value_x + y: self.scale_value_y + z: self.scale_value_x + origin: self.center + canvas.after: + PopMatrix + ''' + + + class ScaleButton(Button): + scale_value_x = NumericProperty(1) + scale_value_y = NumericProperty(1) + scale_value_z = NumericProperty(1) + + + class Test(App): + def build(self): + return Builder.load_string(KV) + + def change_scale(self, instance_button: Button) -> None: + Animation( + scale_value_x=0.5, + scale_value_y=0.5, + scale_value_z=0.5, + d=0.3, + ).start(instance_button) + + + Test().run() + +KivyMD +------ + +.. code-block:: python + + from kivy.animation import Animation + from kivy.lang import Builder + from kivy.uix.behaviors import ButtonBehavior + + from kivymd.app import MDApp + from kivymd.uix.behaviors import ScaleBehavior + from kivymd.uix.boxlayout import MDBoxLayout + + KV = ''' + MDScreen: + + ScaleBox: + size_hint: .5, .5 + pos_hint: {"center_x": .5, "center_y": .5} + on_release: app.change_scale(self) + md_bg_color: "red" + ''' + + + class ScaleBox(ButtonBehavior, ScaleBehavior, MDBoxLayout): + pass + + + class Test(MDApp): + def build(self): + return Builder.load_string(KV) + + def change_scale(self, instance_button: ScaleBox) -> None: + Animation( + scale_value_x=0.5, + scale_value_y=0.5, + scale_value_z=0.5, + d=0.3, + ).start(instance_button) + + + Test().run() +""" + +__all__ = ("ScaleBehavior",) + +from kivy.lang import Builder +from kivy.properties import NumericProperty + +Builder.load_string( + """ + + canvas.before: + PushMatrix + Scale: + x: self.scale_value_x + y: self.scale_value_y + z: self.scale_value_x + origin: self.center + canvas.after: + PopMatrix +""" +) + + +class ScaleBehavior: + """Base class for controlling the scale of the widget.""" + + scale_value_x = NumericProperty(1) + """ + X-axis value. + + :attr:`scale_value_x` is an :class:`~kivy.properties.NumericProperty` + and defaults to `1`. + """ + + scale_value_y = NumericProperty(1) + """ + Y-axis value. + + :attr:`scale_value_y` is an :class:`~kivy.properties.NumericProperty` + and defaults to `1`. + """ + + scale_value_z = NumericProperty(1) + """ + Z-axis value. + + :attr:`scale_value_z` is an :class:`~kivy.properties.NumericProperty` + and defaults to `1`. + """ diff --git a/sbapp/kivymd/uix/behaviors/stencil_behavior.py b/sbapp/kivymd/uix/behaviors/stencil_behavior.py new file mode 100644 index 0000000..4bb5f6d --- /dev/null +++ b/sbapp/kivymd/uix/behaviors/stencil_behavior.py @@ -0,0 +1,134 @@ +""" +Behaviors/Stencil +================= + +.. versionadded:: 1.1.0 + +Base class for controlling the stencil instructions of the widget. + +.. note:: See `Stencil instructions + `_ + for more information. + +Kivy +---- + +.. code-block:: python + + from kivy.lang import Builder + from kivy.app import App + + KV = ''' + Carousel: + + Button: + size_hint: .9, .8 + pos_hint: {"center_x": .5, "center_y": .5} + + canvas.before: + StencilPush + RoundedRectangle: + pos: root.pos + size: root.size + StencilUse + canvas.after: + StencilUnUse + RoundedRectangle: + pos: root.pos + size: root.size + StencilPop + ''' + + + class Test(App): + def build(self): + return Builder.load_string(KV) + + + Test().run() + +KivyMD +------ + +.. code-block:: python + + from kivy.lang import Builder + + from kivymd.app import MDApp + from kivymd.uix.behaviors import StencilBehavior + from kivymd.uix.fitimage import FitImage + + KV = ''' + #:import os os + #:import images_path kivymd.images_path + + + MDCarousel: + + StencilImage: + size_hint: .9, .8 + pos_hint: {"center_x": .5, "center_y": .5} + source: os.path.join(images_path, "logo", "kivymd-icon-512.png") + ''' + + + class StencilImage(FitImage, StencilBehavior): + pass + + + class Test(MDApp): + def build(self): + return Builder.load_string(KV) + + + Test().run() +""" + +__all__ = ("StencilBehavior",) + +from kivy.lang import Builder +from kivy.properties import VariableListProperty + +Builder.load_string( + """ + + canvas.before: + StencilPush + RoundedRectangle: + pos: root.pos + size: root.size + # FIXME: Sometimes the radius has the value [], which get a + # `GraphicException: Invalid radius value, must be list of tuples/numerics` error + radius: root.radius if root.radius else [0, 0, 0, 0] + StencilUse + canvas.after: + StencilUnUse + RoundedRectangle: + pos: root.pos + size: root.size + # FIXME: Sometimes the radius has the value [], which get a + # `GraphicException: Invalid radius value, must be list of tuples/numerics` error + radius: root.radius if root.radius else [0, 0, 0, 0] + StencilPop +""" +) + + +class StencilBehavior: + """Base class for controlling the stencil instructions of the widget.""" + + radius = VariableListProperty([0], length=4) + """ + Canvas radius. + + .. versionadded:: 1.0.0 + + .. code-block:: python + + # Top left corner slice. + MDWidget: + radius: [25, 0, 0, 0] + + :attr:`radius` is an :class:`~kivy.properties.VariableListProperty` + and defaults to `[0, 0, 0, 0]`. + """ diff --git a/sbapp/kivymd/uix/behaviors/toggle_behavior.py b/sbapp/kivymd/uix/behaviors/toggle_behavior.py index 054849c..c396a29 100644 --- a/sbapp/kivymd/uix/behaviors/toggle_behavior.py +++ b/sbapp/kivymd/uix/behaviors/toggle_behavior.py @@ -14,61 +14,104 @@ example: pass -.. code-block:: python +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp - from kivymd.uix.behaviors.toggle_behavior import MDToggleButton - from kivymd.uix.button import MDRectangleFlatButton + .. code-block:: python - KV = ''' - Screen: + from kivy.lang import Builder - MDBoxLayout: - adaptive_size: True - pos_hint: {"center_x": .5, "center_y": .5} + from kivymd.app import MDApp + from kivymd.uix.behaviors.toggle_behavior import MDToggleButton + from kivymd.uix.button import MDFlatButton - MyToggleButton: - text: "Show ads" - group: "x" + KV = ''' + MDScreen: - MyToggleButton: - text: "Do not show ads" - group: "x" + MDBoxLayout: + adaptive_size: True + spacing: "12dp" + pos_hint: {"center_x": .5, "center_y": .5} - MyToggleButton: - text: "Does not matter" - group: "x" - ''' + MyToggleButton: + text: "Show ads" + group: "x" + + MyToggleButton: + text: "Do not show ads" + group: "x" + + MyToggleButton: + text: "Does not matter" + group: "x" + ''' - class MyToggleButton(MDRectangleFlatButton, MDToggleButton): - def __init__(self, **kwargs): - super().__init__(**kwargs) - self.background_down = self.theme_cls.primary_light + class MyToggleButton(MDFlatButton, MDToggleButton): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.background_down = self.theme_cls.primary_color - class Test(MDApp): - def build(self): - return Builder.load_string(KV) + class Test(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) - Test().run() + Test().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.behaviors.toggle_behavior import MDToggleButton + from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.button import MDFlatButton + from kivymd.uix.screen import MDScreen + + + class MyToggleButton(MDFlatButton, MDToggleButton): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.background_down = self.theme_cls.primary_color + + + class Test(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDBoxLayout( + MyToggleButton( + text="Show ads", + group="x", + ), + MyToggleButton( + text="Do not show ads", + group="x", + ), + MyToggleButton( + text="Does not matter", + group="x", + ), + adaptive_size=True, + spacing="12dp", + pos_hint={"center_x": .5, "center_y": .5}, + ), + ) + ) + + + Test().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/toggle-button-1.gif :align: center -.. code-block:: python - - class MyToggleButton(MDFillRoundFlatButton, MDToggleButton): - def __init__(self, **kwargs): - self.background_down = MDApp.get_running_app().theme_cls.primary_dark - super().__init__(**kwargs) - -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/toggle-button-2.gif - :align: center - You can inherit the ``MyToggleButton`` class only from the following classes ---------------------------------------------------------------------------- @@ -88,6 +131,7 @@ from kivy.properties import BooleanProperty, ColorProperty from kivy.uix.behaviors import ToggleButtonBehavior from kivymd.uix.button import ( + ButtonContentsIconText, MDFillRoundFlatButton, MDFillRoundFlatIconButton, MDFlatButton, @@ -149,7 +193,8 @@ class MDToggleButton(ToggleButtonBehavior): # Do the object inherited from the "supported" buttons? if not issubclass(self.__class__, classinfo): raise ValueError( - f"Class {self.__class__} must be inherited from one of the classes in the list {classinfo}" + f"Class {self.__class__} must be inherited from one of the " + f"classes in the list {classinfo}" ) if ( not self.background_normal @@ -165,10 +210,12 @@ class MDToggleButton(ToggleButtonBehavior): ): self.__is_filled = True self.background_normal = self.theme_cls.primary_color - # If not the background_normal must be the same as the inherited one: + # If not background_normal must be the same as the inherited one. else: - self.background_normal = self.md_bg_color[:] - # If no background_down is setted: + self.background_normal = ( + self.md_bg_color[:] if self.md_bg_color else (0, 0, 0, 0) + ) + # If no background_down is setter. if ( not self.background_down ): # This means that if the value == [] or None will return True. @@ -200,3 +247,6 @@ class MDToggleButton(ToggleButtonBehavior): ): # If the background is transparent, the font color must be the # primary color. self.text_color = self.font_color_normal + + if issubclass(self.__class__, ButtonContentsIconText): + self.icon_color = self.text_color diff --git a/sbapp/kivymd/uix/bottomnavigation/bottomnavigation.kv b/sbapp/kivymd/uix/bottomnavigation/bottomnavigation.kv index 546b2c8..79f726b 100644 --- a/sbapp/kivymd/uix/bottomnavigation/bottomnavigation.kv +++ b/sbapp/kivymd/uix/bottomnavigation/bottomnavigation.kv @@ -1,4 +1,3 @@ -#:import sm kivy.uix.screenmanager #:import STANDARD_INCREMENT kivymd.material_resources.STANDARD_INCREMENT @@ -7,9 +6,9 @@ height: STANDARD_INCREMENT if app.theme_cls.material_style == "M2" else "80dp" - MDScreenManager: + ScreenManager: id: tab_manager - transition: sm.FadeTransition(duration=.2) + transition: root.transition(duration=root.transition_duration) on_current: root.dispatch( \ "on_switch_tabs", \ @@ -96,7 +95,7 @@ radius: [16,] size: root._selected_region_width, dp(32) pos: - self.center_x - self.width - dp(8), \ + self.center_x - root._selected_region_width / 2, \ self.center_y - (dp(16)) MDLabel: diff --git a/sbapp/kivymd/uix/bottomnavigation/bottomnavigation.py b/sbapp/kivymd/uix/bottomnavigation/bottomnavigation.py index 9d3dbeb..d99f54b 100755 --- a/sbapp/kivymd/uix/bottomnavigation/bottomnavigation.py +++ b/sbapp/kivymd/uix/bottomnavigation/bottomnavigation.py @@ -62,59 +62,120 @@ For ease of understanding, this code works like this: Example ------- -.. code-block:: python +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp + .. code-block:: python + + from kivy.lang import Builder + + from kivymd.app import MDApp - class Test(MDApp): + class Test(MDApp): - def build(self): - self.theme_cls.material_style = "M3" - return Builder.load_string( - ''' - MDScreen: + def build(self): + self.theme_cls.material_style = "M3" + self.theme_cls.theme_style = "Dark" + return Builder.load_string( + ''' + MDScreen: - MDBottomNavigation: - panel_color: "#eeeaea" - selected_color_background: "#97ecf8" - text_color_active: 0, 0, 0, 1 + MDBottomNavigation: + #panel_color: "#eeeaea" + selected_color_background: "orange" + text_color_active: "lightgrey" - MDBottomNavigationItem: - name: 'screen 1' - text: 'Mail' - icon: 'gmail' - badge_icon: "numeric-10" + MDBottomNavigationItem: + name: 'screen 1' + text: 'Mail' + icon: 'gmail' + badge_icon: "numeric-10" - MDLabel: - text: 'Mail' - halign: 'center' + MDLabel: + text: 'Mail' + halign: 'center' - MDBottomNavigationItem: - name: 'screen 2' - text: 'Discord' - icon: 'discord' - badge_icon: "numeric-5" + MDBottomNavigationItem: + name: 'screen 2' + text: 'Twitter' + icon: 'twitter' + badge_icon: "numeric-5" - MDLabel: - text: 'Discord' - halign: 'center' + MDLabel: + text: 'Twitter' + halign: 'center' - MDBottomNavigationItem: - name: 'screen 3' - text: 'LinkedIN' - icon: 'linkedin' + MDBottomNavigationItem: + name: 'screen 3' + text: 'LinkedIN' + icon: 'linkedin' - MDLabel: - text: 'LinkedIN' - halign: 'center' - ''' - ) + MDLabel: + text: 'LinkedIN' + halign: 'center' + ''' + ) - Test().run() + Test().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.bottomnavigation import MDBottomNavigation, MDBottomNavigationItem + from kivymd.uix.label import MDLabel + from kivymd.uix.screen import MDScreen + + + class Test(MDApp): + def build(self): + self.theme_cls.material_style = "M3" + self.theme_cls.theme_style = "Dark" + return ( + MDScreen( + MDBottomNavigation( + MDBottomNavigationItem( + MDLabel( + text='Mail', + halign='center', + ), + name='screen 1', + text='Mail', + icon='gmail', + badge_icon="numeric-10", + ), + MDBottomNavigationItem( + MDLabel( + text='Twitter', + halign='center', + ), + name='screen 1', + text='Twitter', + icon='twitter', + badge_icon="numeric-10", + ), + MDBottomNavigationItem( + MDLabel( + text='LinkedIN', + halign='center', + ), + name='screen 1', + text='LinkedIN', + icon='linkedin', + badge_icon="numeric-10", + ), + selected_color_background="orange", + text_color_active="lightgrey", + ) + ) + ) + + + Test().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/bottom-navigation.gif :align: center @@ -192,16 +253,13 @@ from kivy.properties import ( ) from kivy.uix.behaviors import ButtonBehavior from kivy.uix.boxlayout import BoxLayout -from kivy.uix.screenmanager import ScreenManagerException +from kivy.uix.screenmanager import FadeTransition, ScreenManagerException from kivymd import uix_path from kivymd.material_resources import STANDARD_INCREMENT from kivymd.theming import ThemableBehavior, ThemeManager from kivymd.uix.anchorlayout import MDAnchorLayout -from kivymd.uix.behaviors import ( - DeclarativeBehavior, - FakeRectangularElevationBehavior, -) +from kivymd.uix.behaviors import CommonElevationBehavior, DeclarativeBehavior from kivymd.uix.behaviors.backgroundcolor_behavior import ( SpecificBackgroundColorBehavior, ) @@ -413,6 +471,28 @@ class MDBottomNavigationItem(MDTab): def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) + def animate_header( + self, bottom_navigation_object, bottom_navigation_header_object + ) -> None: + if bottom_navigation_object.use_text: + Animation(_label_font_size=sp(12), d=0.1).start( + bottom_navigation_object.previous_tab.header + ) + Animation( + _selected_region_width=0, + t="in_out_sine", + d=0, + ).start(bottom_navigation_header_object) + Animation( + _text_color_normal=bottom_navigation_header_object.text_color_normal + if bottom_navigation_object.previous_tab.header.text_color_normal + != [1, 1, 1, 1] + else self.theme_cls.disabled_hint_text_color, + d=0.1, + ).start(bottom_navigation_object.previous_tab.header) + bottom_navigation_object.previous_tab.header.active = False + self.header.active = True + def on_tab_press(self, *args) -> None: """Called when clicking on a panel item.""" @@ -420,28 +500,13 @@ class MDBottomNavigationItem(MDTab): bottom_navigation_header_object = ( bottom_navigation_object.previous_tab.header ) - bottom_navigation_object.ids.tab_manager.current = self.name if bottom_navigation_object.previous_tab is not self: - if bottom_navigation_object.use_text: - Animation(_label_font_size=sp(12), d=0.1).start( - bottom_navigation_object.previous_tab.header - ) - Animation( - _selected_region_width=0, - t="in_out_sine", - d=0, - ).start(bottom_navigation_header_object) - Animation( - _text_color_normal=bottom_navigation_header_object.text_color_normal - if bottom_navigation_object.previous_tab.header.text_color_normal - != [1, 1, 1, 1] - else self.theme_cls.disabled_hint_text_color, - d=0.1, - ).start(bottom_navigation_object.previous_tab.header) - bottom_navigation_object.previous_tab.header.active = False - self.header.active = True - bottom_navigation_object.previous_tab = self + self.animate_header( + bottom_navigation_object, bottom_navigation_header_object + ) + + super().on_tab_press(*args) def on_disabled( self, instance_bottom_navigation_item, disabled_value: bool @@ -498,6 +563,26 @@ class MDBottomNavigation(DeclarativeBehavior, TabbedPanelBase): .. versionadded:: 1.0.0 """ + transition = ObjectProperty(FadeTransition) + """ + Transition animation of bottom navigation screen manager. + + .. versionadded:: 1.1.0 + + :attr:`transition` is an :class:`~kivy.properties.ObjectProperty` + and defaults to `FadeTransition`. + """ + + transition_duration = NumericProperty(0.2) + """ + Duration animation of bottom navigation screen manager. + + .. versionadded:: 1.1.0 + + :attr:`transition_duration` is an :class:`~kivy.properties.NumericProperty` + and defaults to `0.2`. + """ + text_color_normal = ColorProperty([1, 1, 1, 1]) """ Text color of the label when it is not selected. @@ -772,8 +857,6 @@ class MDBottomNavigation(DeclarativeBehavior, TabbedPanelBase): class MDBottomNavigationBar( - ThemableBehavior, - FakeRectangularElevationBehavior, - MDFloatLayout, + ThemableBehavior, CommonElevationBehavior, MDFloatLayout ): pass diff --git a/sbapp/kivymd/uix/bottomsheet/bottomsheet.py b/sbapp/kivymd/uix/bottomsheet/bottomsheet.py index bf7e98f..522f31b 100755 --- a/sbapp/kivymd/uix/bottomsheet/bottomsheet.py +++ b/sbapp/kivymd/uix/bottomsheet/bottomsheet.py @@ -34,7 +34,7 @@ Usage :class:`~MDListBottomSheet` MDTopAppBar: title: "Example BottomSheet" pos_hint: {"top": 1} - elevation: 10 + elevation: 4 MDRaisedButton: text: "Open list bottom sheet" @@ -94,7 +94,7 @@ which will be used as an icon to the left of the item: MDTopAppBar: title: 'Example BottomSheet' pos_hint: {"top": 1} - elevation: 10 + elevation: 4 MDRaisedButton: text: "Open grid bottom sheet" @@ -180,7 +180,7 @@ which will be used as an icon to the left of the item: MDTopAppBar: title: 'Example BottomSheet' pos_hint: {"top": 1} - elevation: 10 + elevation: 4 MDRaisedButton: text: "Open custom bottom sheet" diff --git a/sbapp/kivymd/uix/boxlayout.py b/sbapp/kivymd/uix/boxlayout.py index 11a640d..6922de7 100644 --- a/sbapp/kivymd/uix/boxlayout.py +++ b/sbapp/kivymd/uix/boxlayout.py @@ -93,6 +93,8 @@ from kivymd.uix.behaviors import DeclarativeBehavior class MDBoxLayout(DeclarativeBehavior, BoxLayout, MDAdaptiveWidget): """ - Box layout class. For more information, see in the + Box layout class. + + For more information, see in the :class:`~kivy.uix.boxlayout.BoxLayout` class documentation. """ diff --git a/sbapp/kivymd/uix/button/__init__.py b/sbapp/kivymd/uix/button/__init__.py index dddf186..b833a4c 100644 --- a/sbapp/kivymd/uix/button/__init__.py +++ b/sbapp/kivymd/uix/button/__init__.py @@ -1,6 +1,7 @@ # NOQA F401 from .button import ( BaseButton, + ButtonContentsIconText, MDFillRoundFlatButton, MDFillRoundFlatIconButton, MDFlatButton, diff --git a/sbapp/kivymd/uix/button/button.kv b/sbapp/kivymd/uix/button/button.kv index c7523a4..4cee339 100644 --- a/sbapp/kivymd/uix/button/button.kv +++ b/sbapp/kivymd/uix/button/button.kv @@ -3,9 +3,9 @@ Clear Color: rgba: - (self._md_bg_color or [0.0, 0.0, 0.0, 0.0]) \ + self._md_bg_color \ if not self.disabled else \ - (self._md_bg_color_disabled or [0.0, 0.0, 0.0, 0.0]) + self._md_bg_color_disabled RoundedRectangle: size: self.size pos: self.pos @@ -13,19 +13,17 @@ radius: [root._radius, ] Color: rgba: - root._line_color or [0.0, 0.0, 0.0, 0.0] \ + root._line_color \ if not root.disabled else \ - ( \ - root._line_color_disabled \ - or self._disabled_color \ - or [0.0, 0.0, 0.0, 0.0] \ - ) + (root._line_color_disabled or self._disabled_color) Line: width: root.line_width rounded_rectangle: - (self.x, self.y, self.width, self.height, \ + ( \ + self.x, self.y, self.width, self.height, \ root._radius, root._radius, root._radius, root._radius, \ - self.height) + self.height \ + ) size_hint: None, None anchor_x: root.halign @@ -33,21 +31,28 @@ _round_rad: [self._radius] * 4 - lbl_txt: lbl_txt width: - max(root._min_width, \ - root.padding[0] + lbl_txt.texture_size[0] + root.padding[2]) + max( \ + root._min_width, \ + root.padding[0] + lbl_txt.texture_size[0] + root.padding[2] \ + ) size_hint_min_x: - max(root._min_width, \ - root.padding[0] + lbl_txt.texture_size[0] + root.padding[2]) + max( \ + root._min_width, \ + root.padding[0] + lbl_txt.texture_size[0] + root.padding[2] \ + ) height: - max(root._min_height, \ - root.padding[1] + lbl_txt.texture_size[1] + root.padding[3]) + max( \ + root._min_height, \ + root.padding[1] + lbl_txt.texture_size[1] + root.padding[3] \ + ) size_hint_min_y: - max(root._min_height, \ - root.padding[1] + lbl_txt.texture_size[1] + root.padding[3]) + max( \ + root._min_height, \ + root.padding[1] + lbl_txt.texture_size[1] + root.padding[3] \ + ) MDLabel: id: lbl_txt @@ -84,7 +89,10 @@ # This is only a temporary fix and does not fix the cause of the error. (root._icon_color if root._icon_color else root.theme_cls.text_color) \ if not root.disabled else \ - root.theme_cls.disabled_hint_text_color + root.theme_cls.disabled_hint_text_color \ + if not root.disabled_color else \ + root.disabled_color + on_icon: if self.icon not in md_icons.keys(): self.size_hint = (1, 1) theme_text_color: root._theme_icon_color @@ -131,7 +139,7 @@ id: box adaptive_size: True padding: 0 - spacing: "4dp" + spacing: "8dp" MDIcon: id: lbl_ic @@ -193,48 +201,21 @@ radius: [self.height / 2] - + theme_text_color: "Custom" md_bg_color: self.theme_cls.primary_color + + + padding_x: "8dp" + padding_y: "8dp" + adaptive_size: True + theme_text_color: "Custom" + canvas.before: - PushMatrix - Rotate: - angle: self._angle - axis: (0, 0, 1) - origin: self.center - canvas.after: - PopMatrix - - -# FIXME: Use :class:`~kivymd.uix.boxlayout.MDBoxLayout` instead -# :class:`~kivy.uix.boxlayout.BoxLayout`. - - size_hint: None, None - padding: "8dp", "4dp", "8dp", "4dp" - height: label.texture_size[1] + self.padding[1] * 2 - width: label.texture_size[0] + self.padding[0] * 2 - elevation: 10 - - # TODO: Use `md_bg_color` and `radius` instead `canvasю - canvas: Color: - rgba: - self.theme_cls.primary_color \ - if not root.bg_color else \ - root.bg_color + rgba: self.bg_color RoundedRectangle: - pos: self.pos size: self.size - radius: [5] - - Label: - id: label - markup: True - text: root.text - size_hint: None, None - size: self.texture_size - color: - root.theme_cls.text_color \ - if not root.text_color else \ - root.text_color + pos: self.pos + radius: self.radius diff --git a/sbapp/kivymd/uix/button/button.py b/sbapp/kivymd/uix/button/button.py index 1ae6c31..0fac0e2 100755 --- a/sbapp/kivymd/uix/button/button.py +++ b/sbapp/kivymd/uix/button/button.py @@ -33,31 +33,62 @@ Components/Button MDIconButton ------------ -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-icon-button.gif +.. tabs:: + + .. tab:: Declarative KV style + + .. code-block:: python + + from kivy.lang import Builder + + from kivymd.app import MDApp + + KV = ''' + MDScreen: + + MDIconButton: + icon: "language-python" + pos_hint: {"center_x": .5, "center_y": .5} + ''' + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) + + + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.button import MDIconButton + from kivymd.uix.screen import MDScreen + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDIconButton( + icon="language-python", + pos_hint={"center_x": 0.5, "center_y": 0.5}, + ) + ) + ) + + + Example().run() + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-icon-button.png :align: center -.. code-block:: python - - from kivy.lang import Builder - - from kivymd.app import MDApp - - KV = ''' - MDScreen: - - MDIconButton: - icon: "language-python" - pos_hint: {"center_x": .5, "center_y": .5} - ''' - - - class Example(MDApp): - def build(self): - return Builder.load_string(KV) - - - Example().run() - The :class:`~MDIconButton.icon` parameter must have the name of the icon from ``kivymd/icon_definitions.py`` file. @@ -66,9 +97,9 @@ You can also use custom icons: .. code-block:: kv MDIconButton: - icon: "data/logo/kivy-icon-256.png" + icon: "kivymd/images/logo/kivymd-icon-256.png" -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-icon-custom-button.gif +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-icon-custom-button.png :align: center By default, :class:`~MDIconButton` button has a size ``(dp(48), dp (48))``. @@ -80,7 +111,7 @@ Use :class:`~BaseButton.icon_size` attribute to resize the button: icon: "android" icon_size: "64sp" -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-icon-button-user-font-size.gif +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-icon-button-user-font-size.png :align: center By default, the color of :class:`~MDIconButton` @@ -143,8 +174,10 @@ Material design style 3 ''' - class TestNavigationDrawer(MDApp): + class Example(MDApp): def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" self.theme_cls.material_style = "M3" return Builder.load_string(KV) @@ -166,26 +199,23 @@ Material design style 3 ) - TestNavigationDrawer().run() + Example().run() -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-floating-action-button-m3.gif +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-floating-action-button-m3.png :align: center .. MDFlatButton: MDFlatButton ------------ -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-flat-button.gif - :align: center - To change the text color of: class:`~MDFlatButton` use the ``text_color`` parameter: .. code-block:: kv MDFlatButton: - text: "MDFLATBUTTON" + text: "MDFlatButton" theme_text_color: "Custom" - text_color: 0, 0, 1, 1 + text_color: "orange" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-flat-button-text-color.png :align: center @@ -195,7 +225,7 @@ Or use markup: .. code-block:: kv MDFlatButton: - text: "[color=#00ffcc]MDFLATBUTTON[/color]" + text: "[color=#00ffcc]MDFlatButton[/color]" To specify the font size and font name, use the parameters as in the usual `Kivy` buttons: @@ -203,7 +233,7 @@ To specify the font size and font name, use the parameters as in the usual .. code-block:: kv MDFlatButton: - text: "MDFLATBUTTON" + text: "MDFlatButton" font_size: "18sp" font_name: "path/to/font" @@ -211,33 +241,29 @@ To specify the font size and font name, use the parameters as in the usual MDRaisedButton -------------- -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-raised-button.gif - :align: center - This button is similar to the :class:`~MDFlatButton` button except that you can set the background color for :class:`~MDRaisedButton`: .. code-block:: kv MDRaisedButton: - text: "MDRAISEDBUTTON" - md_bg_color: 1, 0, 1, 1 + text: "MDRaisedButton" + md_bg_color: "red" +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-raised-button.png + :align: center .. MDRectangleFlatButton: MDRectangleFlatButton --------------------- -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-rectangle-flat-button.gif - :align: center - .. code-block:: kv MDRectangleFlatButton: - text: "MDRECTANGLEFLATBUTTON" + text: "MDRectangleFlatButton" theme_text_color: "Custom" - text_color: 1, 0, 0, 1 - line_color: 0, 0, 1, 1 + text_color: "white" + line_color: "red" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-rectangle-flat-button-md-bg-color.png :align: center @@ -246,9 +272,6 @@ MDRectangleFlatButton MDRectangleFlatIconButton ------------------------- -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-rectangle-flat-icon-button.png - :align: center - Button parameters :class:`~MDRectangleFlatIconButton` are the same as button :class:`~MDRectangleFlatButton`, with the addition of the ``theme_icon_color`` and ``icon_color`` parameters as for :class:`~MDIconButton`. @@ -257,12 +280,12 @@ button :class:`~MDRectangleFlatButton`, with the addition of the MDRectangleFlatIconButton: icon: "android" - text: "MDRECTANGLEFLATICONBUTTON" + text: "MDRectangleFlatIconButton" theme_text_color: "Custom" - text_color: 0, 0, 1, 1 - line_color: 1, 0, 1, 1 + text_color: "white" + line_color: "red" theme_icon_color: "Custom" - icon_color: 1, 0, 0, 1 + icon_color: "orange" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-rectangle-flat-icon-button-custom.png :align: center @@ -279,16 +302,18 @@ Without border class Example(MDApp): def build(self): - screen = MDScreen() - screen.add_widget( - MDRectangleFlatIconButton( - text="MDRectangleFlatIconButton", - icon="language-python", - line_color=(0, 0, 0, 0), - pos_hint={"center_x": .5, "center_y": .5}, + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDRectangleFlatIconButton( + text="MDRectangleFlatIconButton", + icon="language-python", + line_color=(0, 0, 0, 0), + pos_hint={"center_x": .5, "center_y": .5}, + ) ) ) - return screen Example().run() @@ -301,6 +326,9 @@ Without border line_color: 0, 0, 0, 0 pos_hint: {"center_x": .5, "center_y": .5} +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-rectangle-flat-icon-button-without-border.png + :align: center + .. MDRoundFlatButton: MDRoundFlatButton ----------------- @@ -308,8 +336,8 @@ MDRoundFlatButton .. code-block:: kv MDRoundFlatButton: - text: "MDROUNDFLATBUTTON" - text_color: 0, 1, 0, 1 + text: "MDRoundFlatButton" + text_color: "white" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-round-flat-button-text-color.png :align: center @@ -318,9 +346,6 @@ MDRoundFlatButton MDRoundFlatIconButton --------------------- -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-round-flat-icon-button.png - :align: center - Button parameters :class:`~MDRoundFlatIconButton` are the same as button :class:`~MDRoundFlatButton`, with the addition of the ``theme_icon_color`` and ``icon_color`` parameters as for :class:`~MDIconButton`: @@ -328,8 +353,12 @@ button :class:`~MDRoundFlatButton`, with the addition of the .. code-block:: kv MDRoundFlatIconButton: + text: "MDRoundFlatIconButton" icon: "android" - text: "MDROUNDFLATICONBUTTON" + text_color: "white" + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-round-flat-icon-button.png + :align: center .. MDFillRoundFlatButton: MDFillRoundFlatButton @@ -359,14 +388,14 @@ button :class:`~MDRaisedButton`, with the addition of the MDTextButton ------------ -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-text-button.png - :align: center - .. code-block:: kv MDTextButton: - text: "MDTEXTBUTTON" - custom_color: 0, 1, 0, 1 + text: "MDTextButton" + custom_color: "white" + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-text-button.png + :align: center .. MDFloatingActionButtonSpeedDial: MDFloatingActionButtonSpeedDial @@ -398,6 +427,8 @@ MDFloatingActionButtonSpeedDial } def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" return Builder.load_string(KV) @@ -408,30 +439,65 @@ MDFloatingActionButtonSpeedDial Or without KV Language: -.. code-block:: python +.. tabs:: - from kivymd.uix.screen import MDScreen - from kivymd.app import MDApp - from kivymd.uix.button import MDFloatingActionButtonSpeedDial + .. tab:: Imperative python style + + .. code-block:: python + + from kivymd.uix.screen import MDScreen + from kivymd.app import MDApp + from kivymd.uix.button import MDFloatingActionButtonSpeedDial - class Example(MDApp): - data = { - 'Python': 'language-python', - 'PHP': 'language-php', - 'C++': 'language-cpp', - } + class Example(MDApp): + data = { + 'Python': 'language-python', + 'PHP': 'language-php', + 'C++': 'language-cpp', + } - def build(self): - screen = MDScreen() - speed_dial = MDFloatingActionButtonSpeedDial() - speed_dial.data = self.data - speed_dial.root_button_anim = True - screen.add_widget(speed_dial) - return screen + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + screen = MDScreen() + speed_dial = MDFloatingActionButtonSpeedDial() + speed_dial.data = self.data + speed_dial.root_button_anim = True + screen.add_widget(speed_dial) + return screen - Example().run() + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.uix.screen import MDScreen + from kivymd.app import MDApp + from kivymd.uix.button import MDFloatingActionButtonSpeedDial + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDFloatingActionButtonSpeedDial( + data={ + 'Python': 'language-python', + 'PHP': 'language-php', + 'C++': 'language-cpp', + }, + root_button_anim=True, + ) + ) + ) + + + Example().run() You can use various types of animation of labels for buttons on the stack: @@ -443,21 +509,133 @@ You can use various types of animation of labels for buttons on the stack: .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-hint.gif :align: center -You can set your color values ​​for background, text of buttons etc: +You can set your color values for background, text of buttons etc: .. code-block:: kv MDFloatingActionButtonSpeedDial: - bg_hint_color: app.theme_cls.primary_light + hint_animation: True + bg_hint_color: app.theme_cls.primary_dark .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-hint-color.png :align: center -.. seealso:: +Binds to individual buttons +--------------------------- - `See full example `_ +.. tabs:: + + .. tab:: Declarative KV style + + .. code-block:: python + + from kivy.lang import Builder + from kivy.properties import DictProperty + + from kivymd.app import MDApp + + KV = ''' + MDScreen: + + MDFloatingActionButtonSpeedDial: + id: speed_dial + data: app.data + root_button_anim: True + hint_animation: True + ''' + + + class Example(MDApp): + data = DictProperty() + + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + self.data = { + 'Python': 'language-python', + 'JS': [ + 'language-javascript', + "on_press", lambda x: print("pressed JS"), + "on_release", lambda x: print( + "stack_buttons", + self.root.ids.speed_dial.stack_buttons + ) + ], + 'PHP': [ + 'language-php', + "on_press", lambda x: print("pressed PHP"), + "on_release", self.callback + ], + 'C++': [ + 'language-cpp', + "on_press", lambda x: print("pressed C++"), + "on_release", lambda x: self.callback() + ], + } + return Builder.load_string(KV) + + def callback(self, *args): + print(args) + + + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.button import MDFloatingActionButtonSpeedDial + from kivymd.uix.screen import MDScreen + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDFloatingActionButtonSpeedDial( + id="speed_dial", + hint_animation=True, + root_button_anim=True, + ) + ) + ) + + def on_start(self): + data = { + "Python": "language-python", + "JS": [ + "language-javascript", + "on_press", lambda x: print("pressed JS"), + "on_release", lambda x: print( + "stack_buttons", + self.root.ids.speed_dial.stack_buttons + ) + ], + "PHP": [ + "language-php", + "on_press", lambda x: print("pressed PHP"), + "on_release", self.callback + ], + "C++": [ + "language-cpp", + "on_press", lambda x: print("pressed C++"), + "on_release", lambda x: self.callback() + ], + } + self.root.ids.speed_dial.data = data + + def callback(self, *args): + print(args) + + + Example().run() """ +from __future__ import annotations + __all__ = ( "BaseButton", "MDIconButton", @@ -495,8 +673,8 @@ from kivy.properties import ( ) from kivy.uix.anchorlayout import AnchorLayout from kivy.uix.behaviors import ButtonBehavior -from kivy.uix.boxlayout import BoxLayout from kivy.uix.floatlayout import FloatLayout +from kivy.weakproxy import WeakProxy from kivymd import uix_path from kivymd.color_definitions import text_colors @@ -505,9 +683,8 @@ from kivymd.theming import ThemableBehavior from kivymd.uix.behaviors import ( CommonElevationBehavior, DeclarativeBehavior, - FakeRectangularElevationBehavior, RectangularRippleBehavior, - RoundedRectangularElevationBehavior, + RotateBehavior, ) from kivymd.uix.label import MDLabel from kivymd.uix.tooltip import MDTooltip @@ -527,6 +704,62 @@ theme_text_color_options = ( "ContrastParentBackground", ) +# FIXME: If you set a new elevation value for the button +# (press the "Set elevation" button), then disable the button +# (press the "Disabled" button), and then enable the button +# (press the "Undisabled" button), then the previously set elevation value is +# reset to zero. +# In addition, if you set a new elevation value +# (press the "Set elevation" button) and click on the button for which we set +# the elevation value, then the new elevation value will receive the previous +# elevation value. This problem is only related to the buttons. +# For example, there is no such problem for the MDCard widget. + +""" +from kivy.lang import Builder + +from kivymd.app import MDApp + +KV = ''' +MDScreen: + + MDRaisedButton: + size_hint: .5, .5 + id: button + pos_hint: {"center_x": .5, "center_y": .5} + elevation: 0 + + MDBoxLayout: + adaptive_size: True + pos_hint: {"center_x": .5} + spacing: 12 + padding: 12 + + MDRaisedButton: + text: "Set elevation" + pos_hint: {"center_x": .5, "bottom": 1} + on_release: button.elevation = 4 + + MDRaisedButton: + text: "Disabled" + pos_hint: {"center_x": .5, "bottom": 1} + on_release: button.disabled = True + + MDRaisedButton: + text: "Undisabled" + pos_hint: {"center_x": .5, "bottom": 1} + on_release: button.disabled = False +''' + + +class Test(MDApp): + def build(self): + return Builder.load_string(KV) + + +Test().run() +""" + class BaseButton( DeclarativeBehavior, @@ -535,7 +768,12 @@ class BaseButton( ButtonBehavior, AnchorLayout, ): - """Base class for all buttons.""" + """ + Base class for all buttons. + + For more information, see in the + :class:`~kivy.uix.anchorlayout.AnchorLayout` class documentation. + """ padding = VariableListProperty([dp(16), dp(8), dp(16), dp(8)]) """ @@ -621,7 +859,7 @@ class BaseButton( text_color = ColorProperty(None) """ - Button text color in (r, g, b, a) format. + Button text color in (r, g, b, a) or string format. :attr:`text_color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -629,7 +867,7 @@ class BaseButton( icon_color = ColorProperty(None) """ - Button icon color in (r, g, b, a) format. + Button icon color in (r, g, b, a) or string format. :attr:`icon_color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -672,7 +910,7 @@ class BaseButton( line_color = ColorProperty(None) """ - Line color for button border. + Line color in (r, g, b, a) or string format for button border. :attr:`line_color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -680,7 +918,7 @@ class BaseButton( line_color_disabled = ColorProperty(None) """ - Disabled line color for button border. + Disabled line color in (r, g, b, a) or string format for button border. .. versionadded:: 1.0.0 @@ -690,7 +928,7 @@ class BaseButton( md_bg_color = ColorProperty(None) """ - Button background color. + Button background color in (r, g, b, a) or string format. :attr:`md_bg_color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -698,7 +936,8 @@ class BaseButton( md_bg_color_disabled = ColorProperty(None) """ - The background color of the button when the button is disabled. + The background color in (r, g, b, a) or string format of the button when + the button is disabled. :attr:`md_bg_color_disabled` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -706,8 +945,8 @@ class BaseButton( disabled_color = ColorProperty(None) """ - The color of the text and icon when the button is disabled, in the - (r, g, b, a) format. + The color of the text and icon when the button is disabled, + in (r, g, b, a) or string format. .. versionadded:: 1.0.0 @@ -728,11 +967,11 @@ class BaseButton( # Note - _radius must be > 0 to avoid rendering issues. _radius = BoundedNumericProperty(dp(4), min=0.0999, errorvalue=0.1) # Properties used for rendering. - _disabled_color = ColorProperty(None) - _md_bg_color = ColorProperty(None) - _md_bg_color_disabled = ColorProperty(None) - _line_color = ColorProperty(None) - _line_color_disabled = ColorProperty(None) + _disabled_color = ColorProperty([0.0, 0.0, 0.0, 0.0]) + _md_bg_color = ColorProperty([0.0, 0.0, 0.0, 0.0]) + _md_bg_color_disabled = ColorProperty([0.0, 0.0, 0.0, 0.0]) + _line_color = ColorProperty([0.0, 0.0, 0.0, 0.0]) + _line_color_disabled = ColorProperty([0.0, 0.0, 0.0, 0.0]) _theme_text_color = OptionProperty(None, options=theme_text_color_options) _theme_icon_color = OptionProperty(None, options=theme_text_color_options) _text_color = ColorProperty(None) @@ -754,8 +993,8 @@ class BaseButton( _animation_fade_bg = ObjectProperty(None, allownone=True) - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) self.theme_cls.bind( primary_palette=self.set_all_colors, theme_style=self.set_all_colors, @@ -809,14 +1048,14 @@ class BaseButton( """Set all button colours (except text/icons).""" # Set main color - self._md_bg_color = ( + _md_bg_color = ( self.md_bg_color or self._default_md_bg_color or self.theme_cls.primary_color ) # Set disabled color - self._md_bg_color_disabled = ( + _md_bg_color_disabled = ( self.md_bg_color_disabled or ( [sum(self.md_bg_color[0:3]) / 3.0] * 3 @@ -829,14 +1068,14 @@ class BaseButton( ) # Set line color - self._line_color = ( + _line_color = ( self.line_color or self._default_line_color or self.theme_cls.primary_color ) # Set disabled line color - self._line_color_disabled = ( + _line_color_disabled = ( self.line_color_disabled or ( [sum(self.line_color[0:3]) / 3.0] * 3 @@ -848,6 +1087,21 @@ class BaseButton( or self.theme_cls.disabled_primary_color ) + if self.theme_cls.theme_style_switch_animation: + Animation( + _md_bg_color=_md_bg_color, + _md_bg_color_disabled=_md_bg_color_disabled, + _line_color=_line_color, + _line_color_disabled=_line_color_disabled, + d=self.theme_cls.theme_style_switch_animation_duration, + t="linear", + ).start(self) + else: + self._md_bg_color = _md_bg_color + self._md_bg_color_disabled = _md_bg_color_disabled + self._line_color = _line_color + self._line_color_disabled = _line_color_disabled + def set_text_color(self, *args) -> None: """ Set _theme_text_color and _text_color based on defaults and options. @@ -872,7 +1126,9 @@ class BaseButton( """ self._theme_icon_color = ( - self.theme_icon_color or self._default_theme_icon_color + (self.theme_icon_color or self._default_theme_icon_color) + if not self.disabled + else "Custom" ) if self._default_icon_color == "PrimaryHue": default_icon_color = text_colors[self.theme_cls.primary_palette][ @@ -931,6 +1187,10 @@ class BaseButton( return super().on_touch_up(touch) def on_disabled(self, instance_button, disabled_value: bool) -> None: + if hasattr(super(), "on_disabled"): + if self.disabled is True: + Animation.cancel_all(self, "elevation") + super().on_disabled(instance_button, disabled_value) Clock.schedule_once(self.set_disabled_color) @@ -948,30 +1208,20 @@ class ButtonElevationBehaviour(CommonElevationBehavior): _elevation_raised = NumericProperty() _anim_raised = ObjectProperty(None, allownone=True) - _default_elevation = 2 + _default_elevation = 3 def __init__(self, **kwargs): + super().__init__(**kwargs) if self.elevation == 0: self.elevation = self._default_elevation - super().__init__(**kwargs) - self.bind(_radius=self.setter("radius")) - self.on_elevation(self, self.elevation) - - def on_elevation(self, instance_button, elevation_value: int) -> None: - super().on_elevation(instance_button, elevation_value) - self._elevation_raised = self.elevation + 6 + if hasattr(self, "radius"): + self.bind(_radius=self.setter("radius")) + Clock.schedule_once(self.create_anim_raised) self.on_disabled(self, self.disabled) - def on__elevation_raised( - self, instance_button, elevation_value: int - ) -> None: - Animation.cancel_all(self, "_elevation") - self._anim_raised = Animation(_elevation=self._elevation_raised, d=0.15) - - def on_disabled(self, instance_button, disabled_value: bool) -> None: - if self.disabled is True: - Animation.cancel_all(self, "_elevation") - super().on_disabled(instance_button, disabled_value) + def create_anim_raised(self, *args) -> None: + self._elevation_raised = self.elevation + 1.2 + self._anim_raised = Animation(elevation=self.elevation + 1, d=0.15) def on_touch_down(self, touch): if not self.disabled: @@ -994,8 +1244,8 @@ class ButtonElevationBehaviour(CommonElevationBehavior): return super().on_touch_up(touch) def stop_elevation_anim(self): - Animation.cancel_all(self, "_elevation") - self._elevation = self.elevation + Animation.cancel_all(self, "elevation") + self.elevation = self._elevation_raised - 1 class ButtonContentsText: @@ -1059,7 +1309,7 @@ class OldButtonIconMixin: self.theme_icon_color = "Custom" -class MDFlatButton(ButtonContentsText, BaseButton): +class MDFlatButton(BaseButton, ButtonContentsText): """ A flat rectangular button with (by default) no border or background. Text is the default text color. @@ -1080,12 +1330,7 @@ class MDFlatButton(ButtonContentsText, BaseButton): """ -class MDRaisedButton( - FakeRectangularElevationBehavior, - ButtonElevationBehaviour, - ButtonContentsText, - BaseButton, -): +class MDRaisedButton(BaseButton, ButtonElevationBehaviour, ButtonContentsText): """ A flat button with (by default) a primary color fill and matching color text. @@ -1098,8 +1343,14 @@ class MDRaisedButton( _default_theme_text_color = "Custom" _default_text_color = "PrimaryHue" + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.shadow_softness = 8 + self.shadow_offset = (0, 2) + self.shadow_radius = self._radius * 2 -class MDRectangleFlatButton(ButtonContentsText, BaseButton): + +class MDRectangleFlatButton(BaseButton, ButtonContentsText): """ A flat button with (by default) a primary color border and primary color text. @@ -1112,7 +1363,7 @@ class MDRectangleFlatButton(ButtonContentsText, BaseButton): class MDRectangleFlatIconButton( - OldButtonIconMixin, ButtonContentsIconText, BaseButton + BaseButton, OldButtonIconMixin, ButtonContentsIconText ): """ A flat button with (by default) a primary color border, primary color text @@ -1127,7 +1378,7 @@ class MDRectangleFlatIconButton( _default_icon_color = "Primary" -class MDRoundFlatButton(ButtonContentsText, BaseButton): +class MDRoundFlatButton(BaseButton, ButtonContentsText): """ A flat button with (by default) fully rounded corners, a primary color border and primary color text. @@ -1138,15 +1389,13 @@ class MDRoundFlatButton(ButtonContentsText, BaseButton): _default_theme_text_color = "Custom" _default_text_color = "Primary" - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) self.rounded_button = True class MDRoundFlatIconButton( - OldButtonIconMixin, - ButtonContentsIconText, - BaseButton, + BaseButton, OldButtonIconMixin, ButtonContentsIconText ): """ A flat button with (by default) rounded corners, a primary color border, @@ -1160,12 +1409,12 @@ class MDRoundFlatIconButton( _default_text_color = "Primary" _default_icon_color = "Primary" - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) self.rounded_button = True -class MDFillRoundFlatButton(ButtonContentsText, BaseButton): +class MDFillRoundFlatButton(BaseButton, ButtonContentsText): """ A flat button with (by default) rounded corners, a primary color fill and primary color text. @@ -1176,15 +1425,13 @@ class MDFillRoundFlatButton(ButtonContentsText, BaseButton): _default_theme_text_color = "Custom" _default_text_color = "PrimaryHue" - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) self.rounded_button = True class MDFillRoundFlatIconButton( - OldButtonIconMixin, - ButtonContentsIconText, - BaseButton, + BaseButton, OldButtonIconMixin, ButtonContentsIconText ): """ A flat button with (by default) rounded corners, a primary color fill, @@ -1198,12 +1445,12 @@ class MDFillRoundFlatIconButton( _default_text_color = "PrimaryHue" _default_icon_color = "PrimaryHue" - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) self.rounded_button = True -class MDIconButton(OldButtonIconMixin, ButtonContentsIcon, BaseButton): +class MDIconButton(BaseButton, OldButtonIconMixin, ButtonContentsIcon): """A simple rounded icon button.""" icon = StringProperty("checkbox-blank-circle") @@ -1217,8 +1464,8 @@ class MDIconButton(OldButtonIconMixin, ButtonContentsIcon, BaseButton): _min_width = NumericProperty(0) _default_icon_pad = max(dp(48) - sp(24), 0) - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) self.rounded_button = True # FIXME: GraphicException: Invalid width value, must be > 0 self.line_width = 0.001 @@ -1236,11 +1483,7 @@ class MDIconButton(OldButtonIconMixin, ButtonContentsIcon, BaseButton): class MDFloatingActionButton( - OldButtonIconMixin, - RoundedRectangularElevationBehavior, - ButtonElevationBehaviour, - ButtonContentsIcon, - BaseButton, + BaseButton, OldButtonIconMixin, ButtonElevationBehaviour, ButtonContentsIcon ): """ Implementation @@ -1268,12 +1511,11 @@ class MDFloatingActionButton( _default_theme_icon_color = "Custom" _default_icon_color = "PrimaryHue" - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) # FIXME: GraphicException: Invalid width value, must be > 0 self.line_width = 0.001 - self.theme_cls.bind(material_style=self.set_size) - self.theme_cls.bind(material_style=self.set__radius) + self.theme_cls.bind(material_style=self.set_size_and_radius) Clock.schedule_once(self.set_size) Clock.schedule_once(self.set__radius) Clock.schedule_once(self.set_font_size) @@ -1287,9 +1529,13 @@ class MDFloatingActionButton( def set__radius(self, *args) -> None: if self.theme_cls.material_style == "M2": + self.shadow_radius = self.height / 2 self.rounded_button = True else: + self.shadow_softness = 8 + self.shadow_offset = (0, 2) self.rounded_button = False + if self.type == "small": self._radius = dp(12) elif self.type == "standard": @@ -1297,6 +1543,12 @@ class MDFloatingActionButton( elif self.type == "large": self._radius = dp(28) + self.shadow_radius = self._radius + + def set_size_and_radius(self, *args) -> None: + self.set_size(args) + self.set__radius(args) + def set_size(self, *args) -> None: if self.theme_cls.material_style == "M2": self.size = dp(56), dp(56) @@ -1316,7 +1568,7 @@ class MDFloatingActionButton( class MDTextButton(ButtonBehavior, MDLabel): color = ColorProperty(None) """ - Button color in (r, g, b, a) format. + Button color in (r, g, b, a) or string format. :attr:`color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -1324,7 +1576,7 @@ class MDTextButton(ButtonBehavior, MDLabel): color_disabled = ColorProperty(None) """ - Button color disabled in (r, g, b, a) format. + Button color disabled in (r, g, b, a) or string format. :attr:`color_disabled` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -1357,14 +1609,6 @@ class MDTextButton(ButtonBehavior, MDLabel): # SpeedDial classes -class BaseFloatingRootButton(MDFloatingActionButton): - _angle = NumericProperty(0) - - def __init__(self, **kwargs): - super().__init__(**kwargs) - self.elevation = 5 - - class BaseFloatingBottomButton(MDFloatingActionButton, MDTooltip): _canvas_width = NumericProperty(0) _padding_right = NumericProperty(0) @@ -1375,41 +1619,48 @@ class BaseFloatingBottomButton(MDFloatingActionButton, MDTooltip): self.height = "46dp" -# FIXME: Use :class:`~kivymd.uix.boxlayout.MDBoxLayout` instead -# :class:`~kivy.uix.boxlayout.BoxLayout`. -class BaseFloatingLabel( - ThemableBehavior, FakeRectangularElevationBehavior, BoxLayout -): - text = StringProperty() - text_color = ColorProperty(None) - bg_color = ColorProperty(None) - - class MDFloatingBottomButton(BaseFloatingBottomButton): - pass + _bg_color = ColorProperty(None) -class MDFloatingRootButton(BaseFloatingRootButton): - pass +class MDFloatingRootButton(RotateBehavior, MDFloatingActionButton): + rotate_value_angle = NumericProperty(0) -class MDFloatingLabel(BaseFloatingLabel): - pass +class MDFloatingLabel(MDLabel): + bg_color = ColorProperty([0, 0, 0, 0]) -class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): +class MDFloatingActionButtonSpeedDial( + DeclarativeBehavior, ThemableBehavior, FloatLayout +): """ + For more information, see in the + :class:`~kivy.uix.floatlayout.FloatLayout` class documentation. + :Events: :attr:`on_open` Called when a stack is opened. :attr:`on_close` Called when a stack is closed. + :attr:`on_press_stack_button` + Called at the on_press event for the stack button. + :attr:`on_release_stack_button` + Called at the on_press event for the stack button. """ icon = StringProperty("plus") """ Root button icon name. + .. code-block:: kv + + MDFloatingActionButtonSpeedDial: + icon: "pencil" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-icon.png + :align: center + :attr:`icon` is a :class:`~kivy.properties.StringProperty` and defaults to `'plus'`. """ @@ -1422,36 +1673,59 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): and defaults to `'right'`. """ - callback = ObjectProperty(lambda x: None) + label_text_color = ColorProperty(None) """ - Custom callback. + Color of floating text labels in (r, g, b, a) or string format. .. code-block:: kv MDFloatingActionButtonSpeedDial: - callback: app.callback + label_text_color: "orange" - .. code-block:: python + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-label-text-color.png + :align: center - def callback(self, instance): - print(instance.icon) - - - :attr:`callback` is a :class:`~kivy.properties.ObjectProperty` + :attr:`label_text_color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. """ - label_text_color = ColorProperty([0, 0, 0, 1]) + label_bg_color = ColorProperty([0, 0, 0, 0]) """ - Floating text color in (r, g, b, a) format. + Background color of floating text labels in (r, g, b, a) or string format. - :attr:`label_text_color` is a :class:`~kivy.properties.ColorProperty` - and defaults to `[0, 0, 0, 1]`. + .. code-block:: kv + + MDFloatingActionButtonSpeedDial: + label_text_color: "black" + label_bg_color: "orange" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-label-bg-color.png + :align: center + + :attr:`label_bg_color` is a :class:`~kivy.properties.ColorProperty` + and defaults to `[0, 0, 0, 0]`. + """ + + label_radius = VariableListProperty([0], length=4) + """ + The radius of the background of floating text labels. + + .. code-block:: kv + + MDFloatingActionButtonSpeedDial: + label_text_color: "black" + label_bg_color: "orange" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-label-radius.png + :align: center + + :attr:`label_radius` is a :class:`~kivy.properties.ColorProperty` + and defaults to `[0, 0, 0, 0]`. """ data = DictProperty() """ - Must be a dictionary + Must be a dictionary. .. code-block:: python @@ -1462,18 +1736,33 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): } """ - right_pad = BooleanProperty(True) + right_pad = BooleanProperty(False) """ - If `True`, the button will increase on the right side by 2.5 pixels - if the :attr:`~hint_animation` parameter equal to `True`. + If `True`, the background for the floating text label will increase by the + number of pixels specified in the :attr:`~right_pad_value` parameter. + + Works only if the :attr:`~hint_animation` parameter is set to `True`. .. rubric:: False + .. code-block:: kv + + MDFloatingActionButtonSpeedDial: + hint_animation: True + right_pad: False + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-right-pad.gif :align: center .. rubric:: True + .. code-block:: kv + + MDFloatingActionButtonSpeedDial: + hint_animation: True + right_pad: True + right_pad_value: "10dp" + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-right-pad-true.gif :align: center @@ -1481,6 +1770,14 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): and defaults to `False`. """ + right_pad_value = NumericProperty(0) + """ + See :attr:`~right_pad` parameter for more information. + + :attr:`right_pad_value` is a :class:`~kivy.properties.NumericProperty` + and defaults to `0`. + """ + root_button_anim = BooleanProperty(False) """ If ``True`` then the root button will rotate 45 degrees when the stack @@ -1569,39 +1866,87 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): bg_color_root_button = ColorProperty(None) """ - Root button color in (r, g, b, a) format. + Background color of root button in (r, g, b, a) or string format. + + .. code-clock:: kv + + MDFloatingActionButtonSpeedDial: + bg_color_root_button: "red" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-bg-color-root-button.png + :align: center :attr:`bg_color_root_button` is a :class:`~kivy.properties.ColorProperty` - and defaults to `[]`. + and defaults to `None`. """ bg_color_stack_button = ColorProperty(None) """ - The color of the buttons in the stack (r, g, b, a) format. + Background color of the stack buttons in (r, g, b, a) or string format. + + .. code-clock:: kv + + MDFloatingActionButtonSpeedDial: + bg_color_root_button: "red" + bg_color_stack_button: "red" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-bg-color-stack-button.png + :align: center :attr:`bg_color_stack_button` is a :class:`~kivy.properties.ColorProperty` - and defaults to `[]`. + and defaults to `None`. """ color_icon_stack_button = ColorProperty(None) """ - The color icon of the buttons in the stack (r, g, b, a) format. + The color icon of the stack buttons in (r, g, b, a) or string format. + + .. code-clock:: kv + + MDFloatingActionButtonSpeedDial: + bg_color_root_button: "red" + bg_color_stack_button: "red" + color_icon_stack_button: "white" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-color-icon-stack-button.png + :align: center :attr:`color_icon_stack_button` is a :class:`~kivy.properties.ColorProperty` - and defaults to `[]`. + and defaults to `None`. """ color_icon_root_button = ColorProperty(None) """ - The color icon of the root button (r, g, b, a) format. + The color icon of the root button in (r, g, b, a) or string format. + + .. code-clock:: kv + + MDFloatingActionButtonSpeedDial: + bg_color_root_button: "red" + bg_color_stack_button: "red" + color_icon_stack_button: "white" + color_icon_root_button: self.color_icon_stack_button + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-color-icon-root-button.png + :align: center :attr:`color_icon_root_button` is a :class:`~kivy.properties.ColorProperty` - and defaults to `[]`. + and defaults to `None`. """ bg_hint_color = ColorProperty(None) """ - Background color for the text of the buttons in the stack (r, g, b, a) format. + Background color for the floating text of the buttons in (r, g, b, a) + or string format. + + .. code-clock:: kv + + MDFloatingActionButtonSpeedDial: + bg_hint_color: "red" + hint_animation: True + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDFloatingActionButtonSpeedDial-bg-hint-color.png + :align: center :attr:`bg_hint_color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -1609,12 +1954,14 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): hint_animation = BooleanProperty(False) """ - Whether to use button extension animation to display text labels. + Whether to use button extension animation to display floating text. :attr:`hint_animation` is a :class:`~kivy.properties.BooleanProperty` and defaults to `False`. """ + stack_buttons = DictProperty() + _label_pos_y_set = False _anim_buttons_data = {} _anim_labels_data = {} @@ -1623,6 +1970,8 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): super().__init__(**kwargs) self.register_event_type("on_open") self.register_event_type("on_close") + self.register_event_type("on_press_stack_button") + self.register_event_type("on_release_stack_button") Window.bind(on_resize=self._update_pos_buttons) def on_open(self, *args): @@ -1657,19 +2006,22 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): if self.state == "open": for widget in self.children: if isinstance(widget, MDFloatingLabel) and self.hint_animation: - widget._elevation = 0 Animation.cancel_all(widget) for item in self.data.items(): if widget.text in item: Animation( _canvas_width=widget.width + dp(24), - _padding_right=dp(5) if self.right_pad else 0, + _padding_right=self.right_pad_value + if self.right_pad + else 0, d=self.opening_time, t=self.opening_transition, ).start(instance_button) if ( instance_button.icon == self.data[f"{widget.text}"] + or instance_button.icon + == self.data[f"{widget.text}"][0] ): Animation( opacity=1, @@ -1684,51 +2036,68 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): def on_data(self, instance_speed_dial, data: dict) -> None: """Creates a stack of buttons.""" - # FIXME: Don't know how to fix AttributeError error: - # File "kivymd/uix/button.py", line 1597, in on_data - # self.add_widget(bottom_button) - # File "kivy/uix/floatlayout.py", line 140, in add_widget - # return super(FloatLayout, self).add_widget(widget, index, canvas) - # File "kivy/uix/layout.py", line 97, in add_widget - # return super(Layout, self).add_widget(widget, index, canvas) - # File "kivy/uix/widget.py", line 629, in add_widget - # canvas.add(widget.canvas) - # AttributeError: 'NoneType' object has no attribute 'add' - super().__init__() + def on_data(*args): + # Bottom buttons. + for name, parameters in data.items(): + name_icon = ( + parameters if (type(parameters) is str) else parameters[0] + ) + + bottom_button = MDFloatingBottomButton( + icon=name_icon, + on_enter=self.on_enter, + on_leave=self.on_leave, + opacity=0, + ) + bottom_button.bind( + on_press=lambda x: self.dispatch("on_press_stack_button"), + on_release=lambda x: self.dispatch( + "on_release_stack_button" + ), + ) + + if "on_press" in parameters: + callback = parameters[parameters.index("on_press") + 1] + bottom_button.bind(on_press=callback) + + if "on_release" in parameters: + callback = parameters[parameters.index("on_release") + 1] + bottom_button.bind(on_release=callback) + + self.set_pos_bottom_buttons(bottom_button) + self.add_widget(bottom_button) + self.stack_buttons[name] = WeakProxy(bottom_button) + # Labels. + floating_text = name + if floating_text: + label = MDFloatingLabel(text=floating_text, opacity=0) + label.bg_color = self.label_bg_color + label.radius = self.label_radius + label.text_color = ( + self.label_text_color + if self.label_text_color + else self.theme_cls.text_color + ) + self.add_widget(label) + # Top root button. + root_button = MDFloatingRootButton(on_release=self.open_stack) + root_button.icon = self.icon + self.set_pos_root_button(root_button) + self.add_widget(root_button) + self.clear_widgets() + self.stack_buttons = {} self._anim_buttons_data = {} self._anim_labels_data = {} self._label_pos_y_set = False - - # Bottom buttons. - for name, name_icon in data.items(): - bottom_button = MDFloatingBottomButton( - icon=name_icon, - on_enter=self.on_enter, - on_leave=self.on_leave, - opacity=0, - ) - bottom_button.bind( - on_release=lambda x=bottom_button: self.callback(x) - ) - self.set_pos_bottom_buttons(bottom_button) - self.add_widget(bottom_button) - # Labels. - floating_text = name - if floating_text: - label = MDFloatingLabel(text=floating_text, opacity=0) - label.text_color = self.label_text_color - self.add_widget(label) - # Top root button. - root_button = MDFloatingRootButton(on_release=self.open_stack) - root_button.icon = self.icon - self.set_pos_root_button(root_button) - self.add_widget(root_button) + Clock.schedule_once(on_data) def on_icon(self, instance_speed_dial, name_icon: str) -> None: - self._get_count_widget(MDFloatingRootButton).icon = name_icon + self._set_button_property(MDFloatingRootButton, "icon", name_icon) - def on_label_text_color(self, instance_speed_dial, color: list) -> None: + def on_label_text_color( + self, instance_speed_dial, color: list | str + ) -> None: for widget in self.children: if isinstance(widget, MDFloatingLabel): widget.text_color = color @@ -1736,34 +2105,52 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): def on_color_icon_stack_button( self, instance_speed_dial, color: list ) -> None: - for widget in self.children: - if isinstance(widget, MDFloatingBottomButton): - widget.text_color = color + self._set_button_property(MDFloatingBottomButton, "icon_color", color) def on_hint_animation(self, instance_speed_dial, value: bool) -> None: for widget in self.children: if isinstance(widget, MDFloatingLabel): - widget.bg_color = (0, 0, 0, 0) + widget.md_bg_color = (0, 0, 0, 0) def on_bg_hint_color(self, instance_speed_dial, color: list) -> None: - for widget in self.children: - if isinstance(widget, MDFloatingBottomButton): - widget._bg_color = color + setattr(MDFloatingBottomButton, "_bg_color", color) def on_color_icon_root_button( self, instance_speed_dial, color: list ) -> None: - self._get_count_widget(MDFloatingRootButton).text_color = color + self._set_button_property(MDFloatingRootButton, "icon_color", color) def on_bg_color_stack_button( self, instance_speed_dial, color: list ) -> None: - for widget in self.children: - if isinstance(widget, MDFloatingBottomButton): - widget.md_bg_color = color + self._set_button_property(MDFloatingBottomButton, "md_bg_color", color) def on_bg_color_root_button(self, instance_speed_dial, color: list) -> None: - self._get_count_widget(MDFloatingRootButton).md_bg_color = color + self._set_button_property(MDFloatingRootButton, "md_bg_color", color) + + def on_press_stack_button(self, *args) -> None: + """ + Called at the on_press event for the stack button. + + .. code-block:: kv + + MDFloatingActionButtonSpeedDial: + on_press_stack_button: print(*args) + + .. versionadded:: 1.1.0 + """ + + def on_release_stack_button(self, *args) -> None: + """ + Called at the on_release event for the stack button. + + .. code-block:: kv + + MDFloatingActionButtonSpeedDial: + on_release_stack_button: print(*args) + + .. versionadded:: 1.1.0 + """ def set_pos_labels(self, instance_floating_label: MDFloatingLabel) -> None: """ @@ -1784,9 +2171,14 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): Called when the application's root window is resized. """ - if self.anchor == "right": - instance_floating_root_button.y = dp(20) - instance_floating_root_button.x = Window.width - (dp(56) + dp(20)) + def set_pos_root_button(*args): + if self.anchor == "right": + instance_floating_root_button.y = dp(20) + instance_floating_root_button.x = self.parent.width - ( + dp(56) + dp(20) + ) + + Clock.schedule_once(set_pos_root_button) def set_pos_bottom_buttons( self, instance_floating_bottom_button: MDFloatingBottomButton @@ -1817,7 +2209,7 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): if self.state != "open": y = 0 - label_position = dp(56) + label_position = dp(54) anim_buttons_data = {} anim_labels_data = {} @@ -1849,7 +2241,7 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): ): # Rotates the root button 45 degrees. Animation( - _angle=-45, + rotate_value_angle=-45, d=self.opening_time_button_rotation, t=self.opening_transition_button_rotation, ).start(widget) @@ -1908,13 +2300,14 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): opacity=0, ).start(widget) elif isinstance(widget, MDFloatingLabel): - Animation(opacity=0, d=0.1).start(widget) + if widget.opacity > 0: + Animation(opacity=0, d=0.1).start(widget) elif ( isinstance(widget, MDFloatingRootButton) and self.root_button_anim ): Animation( - _angle=0, + rotate_value_angle=0, d=self.closing_time_button_rotation, t=self.closing_transition_button_rotation, ).start(widget) @@ -1931,9 +2324,15 @@ class MDFloatingActionButtonSpeedDial(ThemableBehavior, FloatLayout): elif isinstance(widget, MDFloatingLabel): self.set_pos_labels(widget) - def _get_count_widget(self, instance): - widget = None - for widget in self.children: - if isinstance(widget, instance): - break - return widget + def _set_button_property( + self, instance, property_name: str, property_value: str | list + ): + def set_count_widget(*args): + if self.children: + for widget in self.children: + if isinstance(widget, instance): + setattr(instance, property_name, property_value) + Clock.unschedule(set_count_widget) + break + + Clock.schedule_interval(set_count_widget, 0) diff --git a/sbapp/kivymd/uix/card/card.kv b/sbapp/kivymd/uix/card/card.kv index ac67a51..7ed37bd 100644 --- a/sbapp/kivymd/uix/card/card.kv +++ b/sbapp/kivymd/uix/card/card.kv @@ -2,17 +2,6 @@ md_bg_color: app.theme_cls.divider_color - - canvas.before: - Color: - rgba: self.md_bg_color - RoundedRectangle: - size: self.size - pos: self.pos - radius: root.radius - source: root.background - - md_bg_color: self.theme_cls.divider_color \ diff --git a/sbapp/kivymd/uix/card/card.py b/sbapp/kivymd/uix/card/card.py index 53e18c0..d224ae1 100755 --- a/sbapp/kivymd/uix/card/card.py +++ b/sbapp/kivymd/uix/card/card.py @@ -26,26 +26,6 @@ Components/Card MDCard ------ -.. warning:: Starting from the KivyMD 1.1.0 library version, it is necessary - to manually inherit the card class from one of the ``Elevation`` classes - from ``kivymd/uix/behaviors/elevation.py`` module to draw the card shadow. - -.. code-block:: python - - from kivymd.uix.behaviors import RoundedRectangularElevationBehavior - from kivymd.uix.card import MDCard - - - class MD3Card(MDCard, RoundedRectangularElevationBehavior): - '''Implements a material design v3 card.''' - -It actually allows for better control over the providers that implement the -rendering of the shadows. - -.. note:: You can read more information about the classes that implement the - rendering of shadows on this - `documentation page `_. - An example of the implementation of a card in the style of material design version 3 ------------------------------------------------------------------------------------ @@ -59,7 +39,6 @@ An example of the implementation of a card in the style of material design versi from kivy.properties import StringProperty from kivymd.app import MDApp - from kivymd.uix.behaviors import RoundedRectangularElevationBehavior from kivymd.uix.card import MDCard KV = ''' @@ -93,7 +72,7 @@ An example of the implementation of a card in the style of material design versi ''' - class MD3Card(MDCard, RoundedRectangularElevationBehavior): + class MD3Card(MDCard): '''Implements a material design v3 card.''' text = StringProperty() @@ -126,7 +105,6 @@ An example of the implementation of a card in the style of material design versi .. code-block:: python from kivymd.app import MDApp - from kivymd.uix.behaviors import RoundedRectangularElevationBehavior from kivymd.uix.boxlayout import MDBoxLayout from kivymd.uix.button import MDIconButton from kivymd.uix.card import MDCard @@ -135,7 +113,7 @@ An example of the implementation of a card in the style of material design versi from kivymd.uix.screen import MDScreen - class MD3Card(MDCard, RoundedRectangularElevationBehavior): + class MD3Card(MDCard): '''Implements a material design v3 card.''' @@ -170,7 +148,6 @@ An example of the implementation of a card in the style of material design versi adaptive_size=True, color="grey", pos=("12dp", "12dp"), - bold=True, ), ), line_color=(0.2, 0.2, 0.2, 0.8), @@ -255,10 +232,9 @@ End full code MDBoxLayout: orientation: "vertical" - spacing: "10dp" MDTopAppBar: - elevation: 10 + elevation: 4 title: "MDCardSwipe" MDScrollView: @@ -286,7 +262,7 @@ End full code '''Creates a list of cards.''' for i in range(20): - self.screen.ids.md_list.add_widget( + self.root.ids.md_list.add_widget( SwipeToDeleteItem(text=f"One-line item {i}") ) @@ -316,7 +292,7 @@ End full code MDScreen( MDBoxLayout( MDTopAppBar( - elevation=10, + elevation=4, title="MDCardSwipe", ), MDScrollView( @@ -328,7 +304,6 @@ End full code ), id="box", orientation="vertical", - spacing="10dp", ), ) ) @@ -426,7 +401,7 @@ You can use this event to remove items from a list: .. code-block:: python def on_swipe_complete(self, instance): - self.screen.ids.md_list.remove_widget(instance) + self.root.ids.md_list.remove_widget(instance) .. tab:: Decralative python styles @@ -496,10 +471,9 @@ End full code MDBoxLayout: orientation: "vertical" - spacing: "10dp" MDTopAppBar: - elevation: 10 + elevation: 4 title: "MDCardSwipe" MDScrollView: @@ -560,7 +534,7 @@ End full code MDScreen( MDBoxLayout( MDTopAppBar( - elevation=10, + elevation=4, title="MDCardSwipe", ), MDScrollView( @@ -572,7 +546,6 @@ End full code ), id="box", orientation="vertical", - spacing="10dp", ), ) ) @@ -630,26 +603,21 @@ Focus behavior from kivy.lang import Builder from kivymd.app import MDApp - from kivymd.uix.behaviors import FakeRectangularElevationBehavior - from kivymd.uix.card import MDCard KV = ''' MDScreen: - ElevationCard: + MDCard: size_hint: .7, .4 focus_behavior: True pos_hint: {"center_x": .5, "center_y": .5} md_bg_color: "darkgrey" unfocus_color: "darkgrey" focus_color: "grey" + elevation: 6 ''' - class ElevationCard(MDCard, FakeRectangularElevationBehavior): - pass - - class Example(MDApp): def build(self): self.theme_cls.theme_style = "Dark" @@ -663,27 +631,23 @@ Focus behavior .. code-block:: python from kivymd.app import MDApp - from kivymd.uix.behaviors import FakeRectangularElevationBehavior from kivymd.uix.card import MDCard from kivymd.uix.screen import MDScreen - class ElevationCard(MDCard, FakeRectangularElevationBehavior): - pass - - class Example(MDApp): def build(self): self.theme_cls.theme_style = "Dark" return ( MDScreen( - ElevationCard( + MDCard( size_hint=(0.7, 0.4), focus_behavior=True, pos_hint={"center_x": 0.5, "center_y": 0.5}, md_bg_color="darkgrey", unfocus_color="darkgrey", focus_color="grey", + elevation=6, ), ) ) @@ -691,7 +655,6 @@ Focus behavior Example().run() - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/card-focus.gif :align: center @@ -732,12 +695,14 @@ from kivy.properties import ( VariableListProperty, ) from kivy.uix.boxlayout import BoxLayout +from kivy.utils import get_color_from_hex from kivymd import uix_path from kivymd.color_definitions import colors from kivymd.theming import ThemableBehavior from kivymd.uix.behaviors import ( BackgroundColorBehavior, + CommonElevationBehavior, DeclarativeBehavior, RectangularRippleBehavior, ) @@ -781,6 +746,7 @@ class MDCard( ThemableBehavior, BackgroundColorBehavior, RectangularRippleBehavior, + CommonElevationBehavior, FocusBehavior, BoxLayout, ): @@ -800,14 +766,6 @@ class MDCard( and defaults to `False`. """ - elevation = NumericProperty(None, allownone=True) - """ - Elevation value. - - :attr:`elevation` is an :class:`~kivy.properties.NumericProperty` - and defaults to 1. - """ - radius = VariableListProperty([dp(6), dp(6), dp(6), dp(6)]) """ Card radius by default. @@ -831,15 +789,16 @@ class MDCard( """ _bg_color_map = ( - colors["Light"]["CardsDialogs"], - colors["Dark"]["CardsDialogs"], + get_color_from_hex(colors["Light"]["CardsDialogs"]), + get_color_from_hex(colors["Dark"]["CardsDialogs"]), [1.0, 1.0, 1.0, 0.0], ) def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) - self.theme_cls.bind(theme_style=self.update_md_bg_color) - self.theme_cls.bind(material_style=self.set_style) + self.theme_cls.bind( + material_style=self.set_style, theme_style=self.update_md_bg_color + ) Clock.schedule_once(self.set_style) Clock.schedule_once( lambda x: self.on_ripple_behavior(0, self.ripple_behavior) @@ -848,7 +807,9 @@ class MDCard( def update_md_bg_color(self, instance_card, theme_style: str) -> None: if self.md_bg_color in self._bg_color_map: - self.md_bg_color = colors[theme_style]["CardsDialogs"] + self.md_bg_color = get_color_from_hex( + colors[theme_style]["CardsDialogs"] + ) def set_style(self, *args) -> None: self.set_radius() @@ -865,7 +826,7 @@ class MDCard( if self.style == "outlined" or self.style == "filled": self.elevation = 0 elif self.style == "elevated": - self.elevation = 1 + self.elevation = 2 def set_radius(self) -> None: if ( diff --git a/sbapp/kivymd/uix/chip/chip.py b/sbapp/kivymd/uix/chip/chip.py index 433f242..03e1a98 100755 --- a/sbapp/kivymd/uix/chip/chip.py +++ b/sbapp/kivymd/uix/chip/chip.py @@ -132,7 +132,7 @@ Use with elevation icon_right: "close-circle-outline" line_color: app.theme_cls.disabled_hint_text_color md_bg_color: 1, 0, 0, .5 - elevation: 12 + elevation: 4 .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/chip-with-elevation.png :align: center @@ -304,7 +304,6 @@ __all__ = ("MDChip",) import os -from kivy import Logger from kivy.animation import Animation from kivy.lang import Builder from kivy.metrics import dp @@ -314,14 +313,13 @@ from kivy.uix.behaviors import ButtonBehavior from kivymd import uix_path from kivymd.theming import ThemableBehavior from kivymd.uix.behaviors import ( - FakeRectangularElevationBehavior, + CommonElevationBehavior, RectangularRippleBehavior, + ScaleBehavior, TouchBehavior, ) from kivymd.uix.boxlayout import MDBoxLayout from kivymd.uix.label import MDIcon -from kivymd.uix.stacklayout import MDStackLayout -from kivymd.uix.templates import ScaleWidget with open( os.path.join(uix_path, "chip", "chip.kv"), encoding="utf-8" @@ -330,12 +328,12 @@ with open( class MDChip( + MDBoxLayout, ThemableBehavior, RectangularRippleBehavior, - FakeRectangularElevationBehavior, - TouchBehavior, ButtonBehavior, - MDBoxLayout, + CommonElevationBehavior, + TouchBehavior, ): text = StringProperty() """ @@ -456,7 +454,7 @@ class MDChip( self.active = False -class MDScalableCheckIcon(MDIcon, ScaleWidget): +class MDScalableCheckIcon(MDIcon, ScaleBehavior): pos_hint = {"center_y": 0.5} diff --git a/sbapp/kivymd/uix/datatables/datatables.kv b/sbapp/kivymd/uix/datatables/datatables.kv index 188ddde..a6a7220 100644 --- a/sbapp/kivymd/uix/datatables/datatables.kv +++ b/sbapp/kivymd/uix/datatables/datatables.kv @@ -1,5 +1,4 @@ #:import DEVICE_TYPE kivymd.material_resources.DEVICE_TYPE -#:import FakeRectangularElevationBehavior kivymd.uix.behaviors.FakeRectangularElevationBehavior @@ -66,7 +65,7 @@ size_hint_y: None height: self.minimum_height spacing: "4dp" - tooltip_text: root.text + tooltip_text: root.tooltip if root.tooltip else root.text BoxLayout: id: box @@ -175,7 +174,11 @@ font_size: "14sp" on_release: root.table_data.open_pagination_menu() text: - f"{root.table_data.rows_num if root.table_data.rows_num < len(root.table_data.row_data) else len(root.table_data.row_data)}" + "{}".format( \ + root.table_data.rows_num \ + if root.table_data.rows_num < len(root.table_data.row_data) else \ + len(root.table_data.row_data) \ + ) Widget: size_hint_x: None @@ -192,9 +195,11 @@ if root.theme_cls.theme_style == "Dark" else \ (0, 0, 0, 1) text: - f"1-" \ - f"{root.table_data.rows_num if root.table_data.rows_num > len(root.table_data.row_data) else len(root.table_data.row_data)} " \ - f"of {len(root.table_data.row_data)}" + "1-{} of {}".format( \ + root.table_data.rows_num \ + if root.table_data.rows_num > len(root.table_data.row_data) else \ + len(root.table_data.row_data), len(root.table_data.row_data) \ + ) MDIconButton: id: button_back @@ -217,7 +222,7 @@ on_release: root.table_data.set_next_row_data_parts("forward") - + diff --git a/sbapp/kivymd/uix/datatables/datatables.py b/sbapp/kivymd/uix/datatables/datatables.py index b574d06..1626283 100644 --- a/sbapp/kivymd/uix/datatables/datatables.py +++ b/sbapp/kivymd/uix/datatables/datatables.py @@ -11,19 +11,6 @@ Components/DataTables .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-previous.png :align: center -Warnings ---------- - -.. warning:: Data tables are still far from perfect. The class is in constant - change, because of optimizations and bug fixes. If you find a bug or have - an improvement you want to share, take some time and share your discoveries - with us over the main git repo. - Any help is well appreciated. - -.. warning:: In versions prior to `Kivy 2.1.0-dev0` exists an error in which is - the table has only one row in the current page, the table will only render - one column instead of the whole row. - .. note:: `MDDataTable` allows developers to sort the data provided by column. This happens thanks to the use of an external function that you can bind while you're defining the table columns. Be aware that the sorting function @@ -159,6 +146,15 @@ class CellHeader(MDTooltip, BoxLayout): and defaults to `''`. """ + tooltip = StringProperty() + """ + Tooltip containing descriptive text for the column. + If the tooltip is not provided, column `text` shall be used instead. + + :attr:`tooltip` is a :class:`~kivy.properties.StringProperty` + and defaults to `''`. + """ + # TODO: Added example. sort_action = ObjectProperty() """ @@ -340,11 +336,19 @@ class TableHeader(ThemableBehavior, ScrollView): CellHeader( text=col_heading[0], sort_action=col_heading[2], + tooltip=col_heading[3], width=self.cols_minimum[i], table_data=self.table_data, is_sorted=(col_heading[0] == self.sorted_on), sorted_order=self.sorted_order, ) + if len(col_heading) == 4 + else CellHeader( + text=col_heading[0], + sort_action=col_heading[2], + width=self.cols_minimum[i], + table_data=self.table_data, + ) if len(col_heading) == 3 else CellHeader( text=col_heading[0], @@ -356,6 +360,9 @@ class TableHeader(ThemableBehavior, ScrollView): else: # Sets the text in the first cell. self.ids.first_cell.text = col_heading[0] + self.ids.first_cell.tooltip = ( + col_heading[3] if len(col_heading) == 4 else "" + ) self.ids.first_cell.ids.separator.height = 0 self.ids.first_cell.width = self.cols_minimum[i] @@ -765,6 +772,9 @@ class TablePagination(ThemableBehavior, MDBoxLayout): class MDDataTable(ThemableBehavior, AnchorLayout): """ + See :class:`~kivy.uix.anchorlayout.AnchorLayout` class documentation for + more information. + :Events: :attr:`on_row_press` Called when a table row is clicked. @@ -775,7 +785,6 @@ class MDDataTable(ThemableBehavior, AnchorLayout): .. code-block:: python - from kivy.metrics import dp from kivymd.app import MDApp @@ -914,38 +923,82 @@ class MDDataTable(ThemableBehavior, AnchorLayout): """ Data for header columns. - .. code-block:: python + .. tabs:: - from kivy.metrics import dp + .. tab:: Imperative python style - from kivymd.app import MDApp - from kivymd.uix.datatables import MDDataTable - from kivy.uix.anchorlayout import AnchorLayout + .. code-block:: python + + from kivy.metrics import dp + + from kivymd.app import MDApp + from kivymd.uix.datatables import MDDataTable + from kivy.uix.anchorlayout import AnchorLayout - class Example(MDApp): - def build(self): - layout = AnchorLayout() - self.data_tables = MDDataTable( - size_hint=(0.7, 0.6), - use_pagination=True, - check=True, - # name column, width column, sorting function column(optional) - column_data=[ - ("No.", dp(30)), - ("Status", dp(30)), - ("Signal Name", dp(60)), - ("Severity", dp(30)), - ("Stage", dp(30)), - ("Schedule", dp(30), lambda *args: print("Sorted using Schedule")), - ("Team Lead", dp(30)), - ], - ) - layout.add_widget(self.data_tables) - return layout + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + + layout = AnchorLayout() + self.data_tables = MDDataTable( + size_hint=(0.7, 0.6), + use_pagination=True, + check=True, + # name column, width column, sorting function column(optional), custom tooltip + column_data=[ + ("No.", dp(30), None, "Custom tooltip"), + ("Status", dp(30)), + ("Signal Name", dp(60)), + ("Severity", dp(30)), + ("Stage", dp(30)), + ("Schedule", dp(30), lambda *args: print("Sorted using Schedule")), + ("Team Lead", dp(30)), + ], + ) + layout.add_widget(self.data_tables) + return layout - Example().run() + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivy.metrics import dp + + from kivymd.app import MDApp + from kivymd.uix.anchorlayout import MDAnchorLayout + from kivymd.uix.datatables import MDDataTable + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return MDAnchorLayout( + MDDataTable( + size_hint=(0.7, 0.6), + use_pagination=True, + check=True, + # name column, width column, sorting function column(optional) + column_data=[ + ("No.", dp(30)), + ("Status", dp(30)), + ("Signal Name", dp(60)), + ("Severity", dp(30)), + ("Stage", dp(30)), + ("Schedule", dp(30), + lambda *args: print("Sorted using Schedule")), + ("Team Lead", dp(30)), + ], + ) + ) + + + Example().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-column-data.png :align: center @@ -1060,6 +1113,9 @@ class MDDataTable(ThemableBehavior, AnchorLayout): class Example(MDApp): def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + layout = AnchorLayout() data_tables = MDDataTable( size_hint=(0.9, 0.6), @@ -1187,7 +1243,7 @@ class MDDataTable(ThemableBehavior, AnchorLayout): """ Use or not use checkboxes for rows. - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-check.gif + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-check.png :align: center :attr:`check` is an :class:`~kivy.properties.BooleanProperty` @@ -1209,6 +1265,9 @@ class MDDataTable(ThemableBehavior, AnchorLayout): class Example(MDApp): def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + layout = AnchorLayout() data_tables = MDDataTable( size_hint=(0.9, 0.6), @@ -1238,19 +1297,19 @@ class MDDataTable(ThemableBehavior, AnchorLayout): and defaults to `False`. """ - elevation = NumericProperty(8) + elevation = NumericProperty(4) """ Table elevation. :attr:`elevation` is an :class:`~kivy.properties.NumericProperty` - and defaults to `8`. + and defaults to `4`. """ rows_num = NumericProperty(5) """ The number of rows displayed on one page of the table. - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-use-pagination.gif + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-use-pagination-rows-num.png :align: center :attr:`rows_num` is an :class:`~kivy.properties.NumericProperty` @@ -1266,7 +1325,7 @@ class MDDataTable(ThemableBehavior, AnchorLayout): .. rubric:: Center - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-menu-pos-center.png + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-menu-pos-top.png :align: center .. rubric:: Auto @@ -1282,11 +1341,6 @@ class MDDataTable(ThemableBehavior, AnchorLayout): """ Menu height for selecting the number of displayed rows. - .. rubric:: 140dp - - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-menu-height-140.png - :align: center - .. rubric:: 240dp .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-menu-height-240.png @@ -1298,7 +1352,7 @@ class MDDataTable(ThemableBehavior, AnchorLayout): background_color = ColorProperty([0, 0, 0, 0]) """ - Background color in the format (r, g, b, a). + Background color in the format (r, g, b, a) or string format. See :attr:`~kivy.uix.modalview.ModalView.background_color`. Use markup strings @@ -1315,6 +1369,9 @@ class MDDataTable(ThemableBehavior, AnchorLayout): class Example(MDApp): def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + layout = AnchorLayout() data_tables = MDDataTable( size_hint=(0.9, 0.6), @@ -1354,7 +1411,8 @@ class MDDataTable(ThemableBehavior, AnchorLayout): background_color_header = ColorProperty(None) """ - Background color for :class:`~TableHeader` class. + Background color in the format (r, g, b, a) or string format for + :class:`~TableHeader` class. .. versionadded:: 1.0.0 @@ -1374,7 +1432,8 @@ class MDDataTable(ThemableBehavior, AnchorLayout): background_color_cell = ColorProperty(None) """ - Background color for :class:`~CellRow` class. + Background color in the format (r, g, b, a) or string format for + :class:`~CellRow` class. .. versionadded:: 1.0.0 @@ -1395,7 +1454,8 @@ class MDDataTable(ThemableBehavior, AnchorLayout): background_color_selected_cell = ColorProperty(None) """ - Background selected color for :class:`~CellRow` class. + Background selected color in the format (r, g, b, a) or string format for + :class:`~CellRow` class. .. versionadded:: 1.0.0 @@ -1408,7 +1468,7 @@ class MDDataTable(ThemableBehavior, AnchorLayout): background_color_selected_cell="e4514f", ) - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-background-color-selected-cell.gif + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/data-tables-background-color-selected-cell.png :align: center :attr:`background_color_selected_cell` is a :class:`~kivy.properties.ColorProperty` and @@ -1503,6 +1563,9 @@ class MDDataTable(ThemableBehavior, AnchorLayout): data_tables = None def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + layout = MDFloatLayout() # root layout # Creating control buttons. button_box = MDBoxLayout( @@ -1604,6 +1667,9 @@ class MDDataTable(ThemableBehavior, AnchorLayout): data_tables = None def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + layout = MDFloatLayout() layout.add_widget( MDRaisedButton( diff --git a/sbapp/kivymd/uix/dialog/dialog.kv b/sbapp/kivymd/uix/dialog/dialog.kv index 1f057a7..96e01a3 100644 --- a/sbapp/kivymd/uix/dialog/dialog.kv +++ b/sbapp/kivymd/uix/dialog/dialog.kv @@ -18,7 +18,11 @@ PopMatrix - + + shadow_color: 0.0, 0.0, 0.0, 0.0 + elevation: 0 + shadow_softness: 0 + shadow_offset: 0, 0 @@ -28,7 +32,6 @@ orientation: "vertical" size_hint_y: None height: self.minimum_height - elevation: 24 padding: "24dp", "24dp", "8dp", "8dp" radius: root.radius md_bg_color: diff --git a/sbapp/kivymd/uix/dialog/dialog.py b/sbapp/kivymd/uix/dialog/dialog.py index 10f4d67..f254f30 100755 --- a/sbapp/kivymd/uix/dialog/dialog.py +++ b/sbapp/kivymd/uix/dialog/dialog.py @@ -38,6 +38,8 @@ Usage dialog = None def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" return Builder.load_string(KV) def show_alert_dialog(self): @@ -87,6 +89,7 @@ from kivy.uix.modalview import ModalView from kivymd import uix_path from kivymd.material_resources import DEVICE_TYPE from kivymd.theming import ThemableBehavior +from kivymd.uix.behaviors import CommonElevationBehavior from kivymd.uix.button import BaseButton from kivymd.uix.card import MDSeparator from kivymd.uix.list import BaseListItem @@ -97,7 +100,40 @@ with open( Builder.load_string(kv_file.read()) -class BaseDialog(ThemableBehavior, ModalView): +class BaseDialog(ThemableBehavior, ModalView, CommonElevationBehavior): + elevation = NumericProperty(3) + """ + See :attr:`kivymd.uix.behaviors.elevation.CommonElevationBehavior.elevation` + attribute for more information. + + .. versionadded:: 1.1.0 + + :attr:`elevation` is an :class:`~kivy.properties.NumericProperty` + and defaults to `3`. + """ + + shadow_softness = NumericProperty(24) + """ + See :attr:`kivymd.uix.behaviors.elevation.CommonElevationBehavior.shadow_softness` + attribute for more information. + + .. versionadded:: 1.1.0 + + :attr:`shadow_softness` is an :class:`~kivy.properties.NumericProperty` + and defaults to `24`. + """ + + shadow_offset = ListProperty((0, 4)) + """ + See :attr:`kivymd.uix.behaviors.elevation.CommonElevationBehavior.shadow_offset` + attribute for more information. + + .. versionadded:: 1.1.0 + + :attr:`shadow_offset` is an :class:`~kivy.properties.ListProperty` + and defaults to `[0, 4]`. + """ + radius = ListProperty([dp(7), dp(7), dp(7), dp(7)]) """ Dialog corners rounding value. @@ -250,21 +286,22 @@ class MDDialog(BaseDialog): class Example(MDApp): dialog = None - def build(self): - return Builder.load_string(KV) + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) - def show_simple_dialog(self): - if not self.dialog: - self.dialog = MDDialog( - title="Set backup account", - type="simple", - items=[ - Item(text="user01@gmail.com", source="user-1.png"), - Item(text="user02@gmail.com", source="user-2.png"), - Item(text="Add account", source="add-icon.png"), - ], - ) - self.dialog.open() + def show_simple_dialog(self): + if not self.dialog: + self.dialog = MDDialog( + title="Set backup account", + type="simple", + items=[ + Item(text="user01@gmail.com", source="kivymd/images/logo/kivymd-icon-128.png"), + Item(text="user02@gmail.com", source="data/logo/kivy-icon-128.png"), + ], + ) + self.dialog.open() Example().run() @@ -317,6 +354,8 @@ class MDDialog(BaseDialog): dialog = None def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" return Builder.load_string(KV) def show_confirmation_dialog(self): @@ -385,71 +424,140 @@ class MDDialog(BaseDialog): """ Custom content class. - .. code-block:: python + .. tabs:: - from kivy.lang import Builder - from kivy.uix.boxlayout import BoxLayout + .. tab:: Declarative KV style - from kivymd.app import MDApp - from kivymd.uix.button import MDFlatButton - from kivymd.uix.dialog import MDDialog + .. code-block:: python - KV = ''' - - orientation: "vertical" - spacing: "12dp" - size_hint_y: None - height: "120dp" + from kivy.lang import Builder + from kivy.uix.boxlayout import BoxLayout - MDTextField: - hint_text: "City" + from kivymd.app import MDApp + from kivymd.uix.button import MDFlatButton + from kivymd.uix.dialog import MDDialog - MDTextField: - hint_text: "Street" + KV = ''' + + orientation: "vertical" + spacing: "12dp" + size_hint_y: None + height: "120dp" + + MDTextField: + hint_text: "City" + + MDTextField: + hint_text: "Street" - MDFloatLayout: + MDFloatLayout: - MDFlatButton: - text: "ALERT DIALOG" - pos_hint: {'center_x': .5, 'center_y': .5} - on_release: app.show_confirmation_dialog() - ''' + MDFlatButton: + text: "ALERT DIALOG" + pos_hint: {'center_x': .5, 'center_y': .5} + on_release: app.show_confirmation_dialog() + ''' - class Content(BoxLayout): - pass + class Content(BoxLayout): + pass - class Example(MDApp): - dialog = None + class Example(MDApp): + dialog = None - def build(self): - return Builder.load_string(KV) + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) - def show_confirmation_dialog(self): - if not self.dialog: - self.dialog = MDDialog( - title="Address:", - type="custom", - content_cls=Content(), - buttons=[ - MDFlatButton( - text="CANCEL", - theme_text_color="Custom", - text_color=self.theme_cls.primary_color, - ), - MDFlatButton( - text="OK", - theme_text_color="Custom", - text_color=self.theme_cls.primary_color, - ), - ], - ) - self.dialog.open() + def show_confirmation_dialog(self): + if not self.dialog: + self.dialog = MDDialog( + title="Address:", + type="custom", + content_cls=Content(), + buttons=[ + MDFlatButton( + text="CANCEL", + theme_text_color="Custom", + text_color=self.theme_cls.primary_color, + ), + MDFlatButton( + text="OK", + theme_text_color="Custom", + text_color=self.theme_cls.primary_color, + ), + ], + ) + self.dialog.open() - Example().run() + Example().run() + + .. tab:: Declarative Python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.button import MDFlatButton + from kivymd.uix.dialog import MDDialog + from kivymd.uix.floatlayout import MDFloatLayout + from kivymd.uix.textfield import MDTextField + + + class Example(MDApp): + dialog = None + + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDFloatLayout( + MDFlatButton( + text="ALERT DIALOG", + pos_hint={'center_x': 0.5, 'center_y': 0.5}, + on_release=self.show_confirmation_dialog, + ) + ) + ) + + def show_confirmation_dialog(self, *args): + if not self.dialog: + self.dialog = MDDialog( + title="Address:", + type="custom", + content_cls=MDBoxLayout( + MDTextField( + hint_text="City", + ), + MDTextField( + hint_text="Street", + ), + orientation="vertical", + spacing="12dp", + size_hint_y=None, + height="120dp", + ), + buttons=[ + MDFlatButton( + text="CANCEL", + theme_text_color="Custom", + text_color=self.theme_cls.primary_color, + ), + MDFlatButton( + text="OK", + theme_text_color="Custom", + text_color=self.theme_cls.primary_color, + ), + ], + ) + self.dialog.open() + + + Example().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/dialog-custom.png :align: center @@ -460,7 +568,7 @@ class MDDialog(BaseDialog): md_bg_color = ColorProperty(None) """ - Background color in the format (r, g, b, a). + Background color in the (r, g, b, a) or string format. :attr:`md_bg_color` is an :class:`~kivy.properties.ColorProperty` and defaults to `None`. diff --git a/sbapp/kivymd/uix/dropdownitem/dropdownitem.kv b/sbapp/kivymd/uix/dropdownitem/dropdownitem.kv index 4e6d163..82960e3 100644 --- a/sbapp/kivymd/uix/dropdownitem/dropdownitem.kv +++ b/sbapp/kivymd/uix/dropdownitem/dropdownitem.kv @@ -1,7 +1,7 @@ <_Triangle>: canvas: Color: - rgba: root.theme_cls.text_color + rgba: app.theme_cls.text_color Triangle: points: [ \ @@ -13,7 +13,8 @@ orientation: "vertical" - adaptive_size: True + size_hint: None, None + size: self.minimum_size spacing: "5dp" padding: "5dp", "5dp", "5dp", 0 diff --git a/sbapp/kivymd/uix/dropdownitem/dropdownitem.py b/sbapp/kivymd/uix/dropdownitem/dropdownitem.py index e8a98fb..3967c5c 100644 --- a/sbapp/kivymd/uix/dropdownitem/dropdownitem.py +++ b/sbapp/kivymd/uix/dropdownitem/dropdownitem.py @@ -15,13 +15,13 @@ Usage from kivymd.app import MDApp KV = ''' - Screen + MDScreen MDDropDownItem: id: drop_item pos_hint: {'center_x': .5, 'center_y': .5} text: 'Item' - on_release: self.set_item("New Item") + on_release: print("Press item") ''' @@ -48,12 +48,12 @@ import os from kivy.lang import Builder from kivy.properties import NumericProperty, StringProperty from kivy.uix.behaviors import ButtonBehavior +from kivy.uix.boxlayout import BoxLayout from kivy.uix.widget import Widget from kivymd import uix_path from kivymd.theming import ThemableBehavior -from kivymd.uix.behaviors import FakeRectangularElevationBehavior -from kivymd.uix.boxlayout import MDBoxLayout +from kivymd.uix.behaviors import DeclarativeBehavior with open( os.path.join(uix_path, "dropdownitem", "dropdownitem.kv"), encoding="utf-8" @@ -61,15 +61,12 @@ with open( Builder.load_string(kv_file.read()) -class _Triangle(ThemableBehavior, Widget): +class _Triangle(Widget): pass class MDDropDownItem( - ThemableBehavior, - FakeRectangularElevationBehavior, - ButtonBehavior, - MDBoxLayout, + DeclarativeBehavior, ThemableBehavior, ButtonBehavior, BoxLayout ): text = StringProperty() """ diff --git a/sbapp/kivymd/uix/filemanager/filemanager.kv b/sbapp/kivymd/uix/filemanager/filemanager.kv index 57ab89b..ad5704f 100644 --- a/sbapp/kivymd/uix/filemanager/filemanager.kv +++ b/sbapp/kivymd/uix/filemanager/filemanager.kv @@ -6,28 +6,29 @@ background_normal: "" background_down: "" dir_or_file_name: "" + icon_color: 0, 0, 0, 0 _selected: False events_callback: lambda x: None orientation: "vertical" ModifiedOneLineIconListItem: text: root.dir_or_file_name + on_release: root.events_callback(root.path, root) bg_color: self.theme_cls.bg_darkest \ - if root._selected else self.theme_cls.bg_normal - on_release: root.events_callback(root.path, root) + if root._selected else \ + self.theme_cls.bg_normal IconLeftWidget: icon: root.icon - theme_text_color: "Custom" - text_color: self.theme_cls.primary_color + theme_icon_color: "Custom" + icon_color: root.icon_color MDSeparator: - size_hint_y: None - height: self.texture_size[1] + adaptive_height: True shorten: True shorten_from: "center" halign: "center" @@ -61,23 +62,6 @@ text: root.name - - anchor_x: "right" - anchor_y: "bottom" - size_hint_y: None - height: dp(56) - padding: dp(10) - - MDFloatingActionButton: - size_hint: None, None - size:dp(56), dp(56) - icon: root.icon - opposite_colors: True - elevation: 8 - on_release: root.callback() - md_bg_color: root.md_bg_color - - md_bg_color: root.theme_cls.bg_normal @@ -90,7 +74,11 @@ title: root.current_path right_action_items: [["close-box", lambda x: root.exit_manager(1)]] left_action_items: [["chevron-left", lambda x: root.back()]] - elevation: 10 + elevation: 3 + md_bg_color: + app.theme_cls.primary_color \ + if not root.background_color_toolbar else \ + root.background_color_toolbar RecycleView: id: rv diff --git a/sbapp/kivymd/uix/filemanager/filemanager.py b/sbapp/kivymd/uix/filemanager/filemanager.py index 3386c03..1b7727e 100755 --- a/sbapp/kivymd/uix/filemanager/filemanager.py +++ b/sbapp/kivymd/uix/filemanager/filemanager.py @@ -9,7 +9,7 @@ Usage .. code-block:: python - path = '/' # path to the directory that will be opened in the file manager + path = os.path.expanduser("~") # path to the directory that will be opened in the file manager file_manager = MDFileManager( exit_manager=self.exit_manager, # function called when the user reaches directory tree root select_path=self.select_path, # function called when selecting a file/directory @@ -19,7 +19,7 @@ Usage .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager.png :align: center -.. warning:: Be careful! To use the `/` path on Android devices, you need +.. warning:: Be careful! To use the `'/'` path on Android devices, you need special permissions. Therefore, you are likely to get an error. Or with ``preview`` mode: @@ -32,7 +32,7 @@ Or with ``preview`` mode: preview=True, ) -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager-previous.png +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager-preview.png :align: center .. warning:: The `preview` mode is intended only for viewing images and will @@ -43,6 +43,8 @@ Example .. code-block:: python + import os + from kivy.core.window import Window from kivy.lang import Builder @@ -53,19 +55,19 @@ Example KV = ''' MDBoxLayout: - orientation: 'vertical' + orientation: "vertical" MDTopAppBar: title: "MDFileManager" - left_action_items: [['menu', lambda x: None]] - elevation: 10 + left_action_items: [["menu", lambda x: None]] + elevation: 3 MDFloatLayout: MDRoundFlatIconButton: text: "Open manager" icon: "folder" - pos_hint: {'center_x': .5, 'center_y': .6} + pos_hint: {"center_x": .5, "center_y": .5} on_release: app.file_manager_open() ''' @@ -76,23 +78,23 @@ Example Window.bind(on_keyboard=self.events) self.manager_open = False self.file_manager = MDFileManager( - exit_manager=self.exit_manager, - select_path=self.select_path, - preview=True, + exit_manager=self.exit_manager, select_path=self.select_path ) def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" return Builder.load_string(KV) def file_manager_open(self): - self.file_manager.show('/') # output manager to the screen + self.file_manager.show(os.path.expanduser("~")) # output manager to the screen self.manager_open = True - def select_path(self, path): - '''It will be called when you click on the file name + def select_path(self, path: str): + ''' + It will be called when you click on the file name or the catalog selection button. - :type path: str; :param path: path to the selected directory or file; ''' @@ -126,6 +128,9 @@ Not tested on `iOS`. def file_manager_open(self): self.file_manager.show_disks() + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager-show-disks.png + :align: center """ __all__ = ("MDFileManager",) @@ -136,6 +141,7 @@ import re from typing import List, Tuple, Union from kivy import platform +from kivy.clock import Clock from kivy.factory import Factory from kivy.lang import Builder from kivy.metrics import dp @@ -148,7 +154,6 @@ from kivy.properties import ( OptionProperty, StringProperty, ) -from kivy.uix.anchorlayout import AnchorLayout from kivy.uix.behaviors import ButtonBehavior from kivy.uix.modalview import ModalView @@ -156,6 +161,7 @@ from kivymd import images_path, uix_path from kivymd.theming import ThemableBehavior from kivymd.uix.behaviors import CircularRippleBehavior from kivymd.uix.boxlayout import MDBoxLayout +from kivymd.uix.button import MDFloatingActionButton from kivymd.uix.fitimage import FitImage from kivymd.uix.list import BaseListItem from kivymd.uix.relativelayout import MDRelativeLayout @@ -167,9 +173,7 @@ with open( class BodyManager(MDBoxLayout): - """ - Base class for folders and files icons. - """ + """Base class for folders and files icons.""" class BodyManagerWithPreview(MDBoxLayout): @@ -182,47 +186,146 @@ class IconButton(CircularRippleBehavior, ButtonBehavior, FitImage): """Folder icons/thumbnails images in ``preview`` mode.""" -class FloatButton(ThemableBehavior, AnchorLayout): - callback = ObjectProperty() - md_bg_color = ColorProperty([1, 1, 1, 1]) - icon = StringProperty() - - def __init__(self, **kwargs): - super().__init__(**kwargs) - self.theme_cls.bind(primary_palette=self.set_md_bg_color) - - def set_md_bg_color(self, *args): - self.md_bg_color = self.theme_cls.primary_color - - class ModifiedOneLineIconListItem(BaseListItem): _txt_left_pad = NumericProperty("72dp") _txt_top_pad = NumericProperty("16dp") _txt_bot_pad = NumericProperty("15dp") _num_lines = 1 - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) self.height = dp(48) -class MDFileManager(ThemableBehavior, MDRelativeLayout): - icon = StringProperty("check") +class MDFileManager(MDRelativeLayout, ThemableBehavior): """ - The icon that will be used on the directory selection button. + Implements a modal dialog with a file manager. + + For more information, see in the + :class:`~kivymd.uix.relativelayout.MDRelativeLayout` class documentation. + + :Events: + `on_pre_open`: + Called before the MDFileManager is opened. + `on_open`: + Called when the MDFileManager is opened. + `on_pre_dismiss`: + Called before the MDFileManager is closed. + `on_dismiss`: + Called when the MDFileManager is closed. + """ + + icon = StringProperty("check", deprecated=True) + """ + Icon that will be used on the directory selection button. + + .. deprecated:: 1.1.0 + Use :attr:`icon_selection_button` instead. :attr:`icon` is an :class:`~kivy.properties.StringProperty` and defaults to `check`. """ + icon_selection_button = StringProperty("check") + """ + Icon that will be used on the directory selection button. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + MDFileManager( + ... + icon_selection_button="pencil", + ) + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager-icon-selection-button.png + :align: center + + :attr:`icon_selection_button` is an :class:`~kivy.properties.StringProperty` + and defaults to `check`. + """ + + background_color_selection_button = ColorProperty(None) + """ + Background color of the current directory/path selection button. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + MDFileManager( + ... + background_color_selection_button="brown", + ) + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager-background-color-selection-button.png + :align: center + + :attr:`background_color_selection_button` is an :class:`~kivy.properties.ColorProperty` + and defaults to `None`. + """ + + background_color_toolbar = ColorProperty(None) + """ + Background color of the file manager toolbar. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + MDFileManager( + ... + background_color_toolbar="brown", + ) + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager-background-color-toolbar.png + :align: center + + :attr:`background_color_toolbar` is an :class:`~kivy.properties.ColorProperty` + and defaults to `None`. + """ + icon_folder = StringProperty(f"{images_path}folder.png") """ - The icon that will be used for folder icons when using ``preview = True``. + Icon that will be used for folder icons when using ``preview = True``. + + .. code-block:: python + + MDFileManager( + ... + preview=True, + icon_folder="path/to/icon.png", + ) + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager-icon-folder.png + :align: center :attr:`icon` is an :class:`~kivy.properties.StringProperty` and defaults to `check`. """ + icon_color = ColorProperty(None) + """ + Color of the folder icon when the :attr:`preview` property is set to False. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + MDFileManager( + ... + preview=False, + icon_color="brown", + ) + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/file-manager-icon-color.png + :align: center + + :attr:`icon_color` is an :class:`~kivy.properties.ColorProperty` + and defaults to `None`. + """ + exit_manager = ObjectProperty(lambda x: None) """ Function called when the user reaches directory tree root. @@ -259,12 +362,12 @@ class MDFileManager(ThemableBehavior, MDRelativeLayout): and defaults to `all`. """ - current_path = StringProperty(os.getcwd()) + current_path = StringProperty(os.path.expanduser("~")) """ Current directory. :attr:`current_path` is an :class:`~kivy.properties.StringProperty` - and defaults to `/`. + and defaults to `os.path.expanduser("~")`. """ use_access = BooleanProperty(True) @@ -295,9 +398,9 @@ class MDFileManager(ThemableBehavior, MDRelativeLayout): "name", options=["nothing", "name", "date", "size", "type"] ) """ - It can take the values 'nothing' 'name' 'date' 'size' 'type' - sorts files by option - By default, sort by name. - Available options are: `'nothing'`, `'name'`, `'date'`, `'size'`, `'type'`. + It can take the values 'nothing' 'name' 'date' 'size' 'type' - sorts files + by option. By default, sort by name. Available options are: + `'nothing'`, `'name'`, `'date'`, `'size'`, `'type'`. :attr:`sort_by` is an :class:`~kivy.properties.OptionProperty` and defaults to `name`. @@ -325,29 +428,33 @@ class MDFileManager(ThemableBehavior, MDRelativeLayout): """ Contains the list of files that are currently selected. - :attr:`selection` is a read-only :class:`~kivy.properties.ListProperty` and - defaults to `[]`. + :attr:`selection` is a read-only :class:`~kivy.properties.ListProperty` + and defaults to `[]`. + """ + + selection_button = ObjectProperty() + """ + The instance of the directory/path selection button. + + .. versionadded:: 1.1.0 + + :attr:`selection_button` is a read-only :class:`~kivy.properties.ObjectProperty` + and defaults to `None`. """ _window_manager = None _window_manager_open = False - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.register_event_type("on_pre_open") + self.register_event_type("on_open") + self.register_event_type("on_pre_dismiss") + self.register_event_type("on_dismiss") + toolbar_label = self.ids.toolbar.children[1].children[0] toolbar_label.font_style = "Subtitle1" - if ( - self.selector == "any" - or self.selector == "multi" - or self.selector == "folder" - ): - self.add_widget( - FloatButton( - callback=self.select_directory_on_press_button, - md_bg_color=self.theme_cls.primary_color, - icon=self.icon, - ) - ) + Clock.schedule_once(self._create_selection_button) if self.preview: self.ext = [".png", ".jpg", ".jpeg"] @@ -400,15 +507,7 @@ class MDFileManager(ThemableBehavior, MDRelativeLayout): } ) self.ids.rv.data = manager_list - - if not self._window_manager: - self._window_manager = ModalView( - size_hint=self.size_hint, auto_dismiss=False - ) - self._window_manager.add_widget(self) - if not self._window_manager_open: - self._window_manager.open() - self._window_manager_open = True + self._show() def show(self, path: str) -> None: """ @@ -474,6 +573,9 @@ class MDFileManager(ThemableBehavior, MDRelativeLayout): "icon": icon, "dir_or_file_name": name, "events_callback": self.select_dir_or_file, + "icon_color": self.theme_cls.primary_color + if not self.icon_color + else self.icon_color, "_selected": False, } ) @@ -488,19 +590,14 @@ class MDFileManager(ThemableBehavior, MDRelativeLayout): "icon": "file-outline", "dir_or_file_name": os.path.split(name)[1], "events_callback": self.select_dir_or_file, + "icon_color": self.theme_cls.primary_color + if not self.icon_color + else self.icon_color, "_selected": False, } ) self.ids.rv.data = manager_list - - if not self._window_manager: - self._window_manager = ModalView( - size_hint=self.size_hint, auto_dismiss=False - ) - self._window_manager.add_widget(self) - if not self._window_manager_open: - self._window_manager.open() - self._window_manager_open = True + self._show() def get_access_string(self, path: str) -> str: access_string = "" @@ -557,7 +654,9 @@ class MDFileManager(ThemableBehavior, MDRelativeLayout): def close(self) -> None: """Closes the file manager window.""" + self.dispatch("on_pre_dismiss") self._window_manager.dismiss() + self.dispatch("on_dismiss") self._window_manager_open = False def select_dir_or_file( @@ -609,6 +708,84 @@ class MDFileManager(ThemableBehavior, MDRelativeLayout): if self.selector == "folder" or self.selector == "any": self.select_path(self.current_path) + def on_icon(self, instance_file_manager, icon_name: str) -> None: + """Called when the :attr:`icon` property is changed.""" + + self.icon_selection_button = icon_name + + def on_background_color_toolbar( + self, instance_file_manager, color: Union[str, list] + ) -> None: + """ + Called when the :attr:`background_color_toolbar` property is changed. + """ + + def on_background_color_toolbar(*args): + self.ids.toolbar.md_bg_color = color + + Clock.schedule_once(on_background_color_toolbar) + + def on_pre_open(self, *args): + """ + Default pre-open event handler. + + .. versionadded:: 1.1.0 + """ + + def on_open(self, *args): + """ + Default open event handler. + + .. versionadded:: 1.1.0 + """ + + def on_pre_dismiss(self, *args): + """ + Default pre-dismiss event handler. + + .. versionadded:: 1.1.0 + """ + + def on_dismiss(self, *args): + """ + Default dismiss event handler. + + .. versionadded:: 1.1.0 + """ + + def _show(self): + if not self._window_manager: + self._window_manager = ModalView( + size_hint=self.size_hint, auto_dismiss=False + ) + self.size_hint = (1, 1) + self._window_manager.add_widget(self) + + if not self._window_manager_open: + self._window_manager.open() + self._window_manager_open = True + + self.dispatch("on_pre_open") + self.dispatch("on_open") + + def _create_selection_button(self, *args): + if ( + self.selector == "any" + or self.selector == "multi" + or self.selector == "folder" + ): + self.selection_button = MDFloatingActionButton( + on_release=self.select_directory_on_press_button, + md_bg_color=self.theme_cls.primary_color + if not self.background_color_selection_button + else self.background_color_selection_button, + icon=self.icon_selection_button, + pos_hint={"right": 0.99}, + y=dp(12), + elevation=0, + ) + self.add_widget(self.selection_button) + def __sort_files(self, files): def sort_by_name(files): files.sort(key=locale.strxfrm) diff --git a/sbapp/kivymd/uix/fitimage/fitimage.py b/sbapp/kivymd/uix/fitimage/fitimage.py index c67cec2..35ded4d 100644 --- a/sbapp/kivymd/uix/fitimage/fitimage.py +++ b/sbapp/kivymd/uix/fitimage/fitimage.py @@ -132,11 +132,11 @@ from kivy.properties import BooleanProperty, ObjectProperty from kivy.uix.image import AsyncImage from kivy.uix.widget import Widget +from kivymd.uix.behaviors import StencilBehavior from kivymd.uix.boxlayout import MDBoxLayout -from kivymd.uix.templates import StencilWidget -class FitImage(MDBoxLayout, StencilWidget): +class FitImage(MDBoxLayout, StencilBehavior): source = ObjectProperty() """ Filename/source of your image. diff --git a/sbapp/kivymd/uix/gridlayout.py b/sbapp/kivymd/uix/gridlayout.py index 7c296b3..84d883c 100644 --- a/sbapp/kivymd/uix/gridlayout.py +++ b/sbapp/kivymd/uix/gridlayout.py @@ -90,4 +90,7 @@ from kivymd.uix.behaviors import DeclarativeBehavior class MDGridLayout(DeclarativeBehavior, GridLayout, MDAdaptiveWidget): - pass + """ + Grid layout class. For more information, see in the + :class:`~kivy.uix.gridlayout.GridLayout` class documentation. + """ diff --git a/sbapp/kivymd/uix/hero.py b/sbapp/kivymd/uix/hero.py index 34eef37..70bb572 100644 --- a/sbapp/kivymd/uix/hero.py +++ b/sbapp/kivymd/uix/hero.py @@ -63,7 +63,7 @@ Base example x: 24 FitImage: - source: "https://github.com/kivymd/internal/raw/main/logo/kivymd_logo_blue.png" + source: "kivymd/images/logo/kivymd-icon-512.png" size_hint: None, None size: hero_from.size @@ -72,7 +72,7 @@ Base example pos_hint: {"center_x": .5} y: "36dp" on_release: - root.current_hero = "hero" + root.current_heroes = ["hero"] root.current = "screen B" MDScreen: @@ -82,6 +82,7 @@ Base example MDHeroTo: id: hero_to + tag: "hero" size_hint: None, None size: "220dp", "220dp" pos_hint: {"center_x": .5, "center_y": .5} @@ -91,7 +92,7 @@ Base example pos_hint: {"center_x": .5} y: "36dp" on_release: - root.current_hero = "hero" + root.current_heroes = ["hero"] root.current = "screen A" ''' @@ -113,6 +114,7 @@ Note that the child of the :class:`~MDHeroFrom` widget must have the size of the MDHeroFrom: id: hero_from + tag: "hero" FitImage: size_hint: None, None @@ -127,7 +129,7 @@ container in which the hero is located: MDRaisedButton: text: "Move Hero To Screen B" on_release: - root.current_hero = "hero" + root.current_heroes = ["hero"] root.current = "screen 2" If you need to switch to a screen that does not contain heroes, set the @@ -138,7 +140,7 @@ If you need to switch to a screen that does not contain heroes, set the MDRaisedButton: text: "Go To Another Screen" on_release: - root.current_hero = "" + root.current_heroes = [] root.current = "another screen" Example @@ -166,7 +168,7 @@ Example x: 24 FitImage: - source: "https://github.com/kivymd/internal/raw/main/logo/kivymd_logo_blue.png" + source: "kivymd/images/logo/kivymd-icon-512.png" size_hint: None, None size: hero_from.size @@ -175,7 +177,7 @@ Example pos_hint: {"center_x": .5} y: "36dp" on_release: - root.current_hero = "hero" + root.current_heroes = ["hero"] root.current = "screen B" MDScreen: @@ -185,6 +187,7 @@ Example MDHeroTo: id: hero_to + tag: "hero" size_hint: None, None size: "220dp", "220dp" pos_hint: {"center_x": .5, "center_y": .5} @@ -194,7 +197,7 @@ Example pos_hint: {"center_x": .5} y: "52dp" on_release: - root.current_hero = "" + root.current_heroes = [] root.current = "screen C" MDRaisedButton: @@ -202,7 +205,7 @@ Example pos_hint: {"center_x": .5} y: "8dp" on_release: - root.current_hero = "hero" + root.current_heroes = ["hero"] root.current = "screen A" MDScreen: @@ -283,7 +286,7 @@ background color of the hero during the flight between the screens: pos_hint: {"center_x": .5} y: "36dp" on_release: - root.current_hero = "hero" + root.current_heroes = ["hero"] root.current = "screen B" MDScreen: @@ -293,6 +296,7 @@ background color of the hero during the flight between the screens: MDHeroTo: id: hero_to + tag: "hero" size_hint: None, None size: "220dp", "220dp" pos_hint: {"center_x": .5, "center_y": .5} @@ -302,7 +306,7 @@ background color of the hero during the flight between the screens: pos_hint: {"center_x": .5} y: "36dp" on_release: - root.current_hero = "hero" + root.current_heroes = ["hero"] root.current = "screen A" ''' @@ -370,7 +374,7 @@ Usage with ScrollView radius: 24 box_radius: 0, 0, 24, 24 box_color: 0, 0, 0, .5 - source: "image.jpg" + source: "kivymd/images/logo/kivymd-icon-512.png" size_hint: None, None size: root.size mipmap: True @@ -399,7 +403,7 @@ Usage with ScrollView MDScreen: name: "screen B" - hero_to: hero_to + heroes_to: [hero_to] MDHeroTo: id: hero_to @@ -412,7 +416,7 @@ Usage with ScrollView pos_hint: {"center_x": .5} y: "36dp" on_release: - root.current_hero = "hero" + root.current_heroes = [hero_to.tag] root.current = "screen A" ''' @@ -441,7 +445,8 @@ Usage with ScrollView def on_release(self): def switch_screen(*args): - self.manager.current_hero = self.tag + self.manager.current_heroes = [self.tag] + self.manager.ids.hero_to.tag = self.tag self.manager.current = "screen B" Clock.schedule_once(switch_screen, 0.2) @@ -465,6 +470,93 @@ Usage with ScrollView .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/hero-usage-with-scrollview.gif :align: center + +Using multiple heroes at the same time +-------------------------------------- + +.. code-block:: python + + from kivy.lang import Builder + + from kivymd.app import MDApp + + KV = ''' + MDScreenManager: + + MDScreen: + name: "screen A" + md_bg_color: "lightblue" + + MDHeroFrom: + id: hero_kivymd + tag: "kivymd" + size_hint: None, None + size: "200dp", "200dp" + pos_hint: {"top": .98} + x: 24 + + FitImage: + source: "kivymd/images/logo/kivymd-icon-512.png" + size_hint: None, None + size: hero_kivymd.size + + MDHeroFrom: + id: hero_kivy + tag: "kivy" + size_hint: None, None + size: "200dp", "200dp" + pos_hint: {"top": .98} + x: 324 + + FitImage: + source: "data/logo/kivy-icon-512.png" + size_hint: None, None + size: hero_kivy.size + + MDRaisedButton: + text: "Move Hero To Screen B" + pos_hint: {"center_x": .5} + y: "36dp" + on_release: + root.current_heroes = ["kivymd", "kivy"] + root.current = "screen B" + + MDScreen: + name: "screen B" + heroes_to: hero_to_kivymd, hero_to_kivy + md_bg_color: "cadetblue" + + MDHeroTo: + id: hero_to_kivy + tag: "kivy" + size_hint: None, None + pos_hint: {"center_x": .5, "center_y": .5} + + MDHeroTo: + id: hero_to_kivymd + tag: "kivymd" + size_hint: None, None + pos_hint: {"right": 1, "top": 1} + + MDRaisedButton: + text: "Move Hero To Screen A" + pos_hint: {"center_x": .5} + y: "36dp" + on_release: + root.current_heroes = ["kivy", "kivymd"] + root.current = "screen A" + ''' + + + class Test(MDApp): + def build(self): + return Builder.load_string(KV) + + + Test().run() + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/hero-multiple-heroes.gif + :align: center """ from kivy.properties import StringProperty @@ -476,6 +568,9 @@ class MDHeroFrom(MDBoxLayout): """ The container from which the hero begins his flight. + For more information, see in the + :class:`~kivymd.uix.boxlayout.MDBoxLayout` class documentation. + :Events: `on_transform_in` when the hero flies from screen **A** to screen **B**. @@ -487,7 +582,7 @@ class MDHeroFrom(MDBoxLayout): """ Tag ID for heroes. - :attr:`shift_right` is an :class:`~kivy.properties.StringProperty` + :attr:`tag` is an :class:`~kivy.properties.StringProperty` and defaults to `''`. """ @@ -504,4 +599,17 @@ class MDHeroFrom(MDBoxLayout): class MDHeroTo(MDBoxLayout): - """The container in which the hero comes.""" + """ + The container in which the hero comes. + + For more information, see in the + :class:`~kivymd.uix.boxlayout.MDBoxLayout` class documentation. + """ + + tag = StringProperty(allownone=True) + """ + Tag ID for heroes. + + :attr:`tag` is an :class:`~kivy.properties.StringProperty` + and defaults to `''`. + """ diff --git a/sbapp/kivymd/uix/imagelist/imagelist.py b/sbapp/kivymd/uix/imagelist/imagelist.py index 310cee3..970ed2e 100755 --- a/sbapp/kivymd/uix/imagelist/imagelist.py +++ b/sbapp/kivymd/uix/imagelist/imagelist.py @@ -65,12 +65,13 @@ Implementation :align: center """ -__all__ = "MDSmartTile" +__all__ = [ + "MDSmartTile", +] import os from kivy.lang import Builder -from kivy.logger import Logger from kivy.properties import ( BooleanProperty, ColorProperty, diff --git a/sbapp/kivymd/uix/label/label.kv b/sbapp/kivymd/uix/label/label.kv index b974336..71cac91 100644 --- a/sbapp/kivymd/uix/label/label.kv +++ b/sbapp/kivymd/uix/label/label.kv @@ -17,8 +17,14 @@ rgba: (1, 1, 1, 1) if self.source else (0, 0, 0, 0) Rectangle: source: self.source if self.source else None - pos: self.pos - size: self.size + pos: + self.pos \ + if not self.source else \ + (self.x - self._size[0] / 2, self.y) + size: + self._size \ + if self.source else \ + self.size font_style: "Icon" text: u"{}".format(md_icons[root.icon]) if root.icon in md_icons else "blank" diff --git a/sbapp/kivymd/uix/label/label.py b/sbapp/kivymd/uix/label/label.py index 9d2a65f..fc34ba4 100755 --- a/sbapp/kivymd/uix/label/label.py +++ b/sbapp/kivymd/uix/label/label.py @@ -222,14 +222,18 @@ __all__ = ("MDLabel", "MDIcon") import os from typing import Union +from kivy.animation import Animation from kivy.clock import Clock +from kivy.graphics import Color, Rectangle from kivy.lang import Builder from kivy.metrics import sp from kivy.properties import ( AliasProperty, BooleanProperty, ColorProperty, + ListProperty, NumericProperty, + ObjectProperty, OptionProperty, StringProperty, ) @@ -322,6 +326,7 @@ class MDLabel(DeclarativeBehavior, ThemableBehavior, Label, MDAdaptiveWidget): parent_background = ColorProperty(None) can_capitalize = BooleanProperty(True) + canvas_bg = ObjectProperty() def __init__(self, **kwargs): super().__init__(**kwargs) @@ -349,6 +354,7 @@ class MDLabel(DeclarativeBehavior, ThemableBehavior, Label, MDAdaptiveWidget): font_info = self.theme_cls.font_styles[self.font_style] self.font_name = font_info[0] self.font_size = sp(font_info[1]) + if font_info[2] and self.can_capitalize: self._capitalizing = True else: @@ -374,29 +380,68 @@ class MDLabel(DeclarativeBehavior, ThemableBehavior, Label, MDAdaptiveWidget): # generic None value it's not yet been set self._text_color_str = "" if theme_text_color == "Custom" and self.text_color: - self.color = self.text_color + color = self.text_color elif ( theme_text_color == "ContrastParentBackground" and self.parent_background ): - self.color = get_contrast_text_color(self.parent_background) + color = get_contrast_text_color(self.parent_background) else: - self.color = [0, 0, 0, 1] + color = [0, 0, 0, 1] - def on_text_color(self, instance_label, color: list) -> None: + if self.theme_cls.theme_style_switch_animation: + Animation( + color=color, + d=self.theme_cls.theme_style_switch_animation_duration, + t="linear", + ).start(self) + else: + self.color = color + + def on_text_color(self, instance_label, color: Union[list, str]) -> None: if self.theme_text_color == "Custom": - self.color = self.text_color + if self.theme_cls.theme_style_switch_animation: + Animation( + color=self.text_color, + d=self.theme_cls.theme_style_switch_animation_duration, + t="linear", + ).start(self) + else: + self.color = self.text_color def on_opposite_colors(self, *args) -> None: self.on_theme_text_color(self, self.theme_text_color) + def on_md_bg_color(self, instance_label, color: Union[list, str]) -> None: + self.canvas.remove_group("Background_instruction") + with self.canvas.before: + Color(rgba=color) + self.canvas_bg = Rectangle(pos=self.pos, size=self.size) + self.bind(pos=self.update_canvas_bg_pos) + + def on_size(self, instance_label, size: list) -> None: + if self.canvas_bg: + self.canvas_bg.size = size + + def update_canvas_bg_pos(self, instance_label, pos: list) -> None: + if self.canvas_bg: + self.canvas_bg.pos = pos + def _do_update_theme_color(self, *args): if self._text_color_str: - self.color = getattr(self.theme_cls, self._text_color_str) if not self.disabled: - self.color = getattr(self.theme_cls, self._text_color_str) + color = getattr(self.theme_cls, self._text_color_str) else: - self.color = getattr(self.theme_cls, "disabled_hint_text_color") + color = getattr(self.theme_cls, "disabled_hint_text_color") + + if self.theme_cls.theme_style_switch_animation: + Animation( + color=color, + d=self.theme_cls.theme_style_switch_animation_duration, + t="linear", + ).start(self) + else: + self.color = color class MDIcon(MDFloatLayout, MDLabel): @@ -456,11 +501,16 @@ class MDIcon(MDFloatLayout, MDLabel): and defaults to `None`. """ - def __init__(self, **kwargs): + _size = ListProperty((0, 0)) + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + Clock.schedule_once(self.adjust_size) + + def adjust_size(self, *args) -> None: from kivymd.uix.selectioncontrol import MDCheckbox - super().__init__(**kwargs) if not isinstance(self, MDCheckbox): self.size_hint = (None, None) - self.size = self.texture_size + self._size = self.texture_size[1], self.texture_size[1] self.adaptive_size = True diff --git a/sbapp/kivymd/uix/list/list.py b/sbapp/kivymd/uix/list/list.py index 5f4443f..9f02cab 100755 --- a/sbapp/kivymd/uix/list/list.py +++ b/sbapp/kivymd/uix/list/list.py @@ -57,6 +57,7 @@ based on the above classes. - OneLineAvatarListItem_ - TwoLineAvatarListItem_ - ThreeLineAvatarListItem_ + - OneLineIconListItem_ - TwoLineIconListItem_ - ThreeLineIconListItem_ @@ -68,35 +69,71 @@ based on the above classes. - TwoLineAvatarIconListItem_ - ThreeLineAvatarIconListItem_ +- OneLineRightIconListItem_ +- TwoLineRightIconListItem_ +- ThreeLineRightIconListItem_ + Usage ----- -.. code-block:: python +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp - from kivymd.uix.list import OneLineListItem + .. code-block:: python - KV = ''' - ScrollView: + from kivy.lang import Builder - MDList: - id: container - ''' + from kivymd.app import MDApp + from kivymd.uix.list import OneLineListItem + + KV = ''' + MDScrollView: + + MDList: + id: container + ''' - class Test(MDApp): - def build(self): - return Builder.load_string(KV) + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + return Builder.load_string(KV) - def on_start(self): - for i in range(20): - self.root.ids.container.add_widget( - OneLineListItem(text=f"Single-line item {i}") - ) + def on_start(self): + for i in range(20): + self.root.ids.container.add_widget( + OneLineListItem(text=f"Single-line item {i}") + ) - Test().run() + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.list import OneLineListItem + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + return ( + MDScrollView( + MDList( + id="container" + ) + ) + ) + + def on_start(self): + for i in range(20): + self.root.ids.container.add_widget( + OneLineListItem(text=f"Single-line item {i}") + ) + + Example().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/lists.gif :align: center @@ -104,37 +141,76 @@ Usage Events of List -------------- -.. code-block:: python +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp + .. code-block:: python - KV = ''' - ScrollView: + from kivy.lang import Builder - MDList: + from kivymd.app import MDApp - OneLineAvatarIconListItem: - on_release: print("Click!") + KV = ''' + MDScrollView: - IconLeftWidget: - icon: "github" + MDList: - OneLineAvatarIconListItem: - on_release: print("Click 2!") + OneLineAvatarIconListItem: + on_release: print("Click!") - IconLeftWidget: - icon: "gitlab" - ''' + IconLeftWidget: + icon: "github" + + OneLineAvatarIconListItem: + on_release: print("Click 2!") + + IconLeftWidget: + icon: "gitlab" + ''' - class MainApp(MDApp): - def build(self): - return Builder.load_string(KV) + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + return Builder.load_string(KV) - MainApp().run() + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.scrollview import MDScrollView + from kivymd.uix.list import MDList, OneLineAvatarIconListItem, IconLeftWidget + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + return ( + MDScrollView( + MDList( + OneLineAvatarIconListItem( + IconLeftWidget( + icon="github" + ), + on_release=lambda x: print("Click!") + ), + OneLineAvatarIconListItem( + IconLeftWidget( + icon="gitlab" + ), + on_release=lambda x: print("Click 2!") + ), + ) + ) + ) + + + Example().run() .. OneLineListItem: OneLineListItem @@ -179,62 +255,220 @@ ThreeLineListItem OneLineAvatarListItem --------------------- -.. code-block:: kv +.. tabs:: - OneLineAvatarListItem: - text: "Single-line item with avatar" + .. tab:: Declarative KV style - ImageLeftWidget: - source: "data/logo/kivy-icon-256.png" + .. code-block:: kv -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/lists-map.png + OneLineAvatarListItem: + text: "Single-line item with avatar" + + ImageLeftWidget: + source: "kivymd/images/logo/kivymd-icon-256.png" + + .. tab:: Declarative python style + + .. code-block:: python + + OneLineAvatarListItem( + ImageLeftWidget( + source="kivymd/images/logo/kivymd-icon-256.png" + ), + text="Single-line item with avatar", + ) + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/OneLineAvatarListItem.png :align: center .. TwoLineAvatarListItem: TwoLineAvatarListItem --------------------- -.. code-block:: kv +.. tabs:: - TwoLineAvatarListItem: - text: "Two-line item with avatar" - secondary_text: "Secondary text here" + .. tab:: Declarative KV style - ImageLeftWidget: - source: "data/logo/kivy-icon-256.png" + .. code-block:: kv + + TwoLineAvatarListItem: + text: "Two-line item with avatar" + secondary_text: "Secondary text here" + + ImageLeftWidget: + source: "kivymd/images/logo/kivymd-icon-256.png" + + .. tab:: Declarative python style + + .. code-block:: python + + OneLineAvatarListItem( + ImageLeftWidget( + source="kivymd/images/logo/kivymd-icon-256.png" + ), + text="Single-line item with avatar", + secondary_text: "Secondary text here", + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/TwoLineAvatarListItem.png :align: center - .. ThreeLineAvatarListItem: ThreeLineAvatarListItem ----------------------- -.. code-block:: kv +.. tabs:: - ThreeLineAvatarListItem: - text: "Three-line item with avatar" - secondary_text: "Secondary text here" - tertiary_text: "fit more text than usual" + .. tab:: Declarative KV style - ImageLeftWidget: - source: "data/logo/kivy-icon-256.png" + .. code-block:: kv + + ThreeLineAvatarListItem: + text: "Three-line item with avatar" + secondary_text: "Secondary text here" + tertiary_text: "fit more text than usual" + + ImageLeftWidget: + source: "kivymd/images/logo/kivymd-icon-256.png" + + .. tab:: Declarative python style + + .. code-block:: python + + OneLineAvatarListItem( + ImageLeftWidget( + source="kivymd/images/logo/kivymd-icon-256.png" + ), + text="Single-line item with avatar", + secondary_text: "Secondary text here", + tertiary_text: "fit more text than usual" + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/ThreeLineAvatarListItem.png :align: center +.. OneLineRightIconListItem: +OneLineRightIconListItem +------------------------ + +.. tabs:: + + .. tab:: Declarative KV style + + .. code-block:: kv + + OneLineRightIconListItem: + text: "Single-line item with avatar" + + ImageRightWidget: + source: "kivymd/images/logo/kivymd-icon-256.png" + + .. tab:: Declarative python style + + .. code-block:: python + + OneLineRightIconListItem( + ImageRightWidget( + source="kivymd/images/logo/kivymd-icon-256.png" + ), + text="Single-line item with avatar", + ) + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/OneLineRightIconListItem.png + :align: center + +.. TwoLineRightIconListItem: +TwoLineRightIconListItem +------------------------ + +.. tabs:: + + .. tab:: Declarative KV style + + .. code-block:: kv + + TwoLineRightIconListItem: + text: "Single-line item with avatar" + secondary_text: "Secondary text here" + + ImageRightWidget: + source: "kivymd/images/logo/kivymd-icon-256.png" + + .. tab:: Declarative python style + + .. code-block:: python + + TwoLineRightIconListItem( + ImageRightWidget( + source="kivymd/images/logo/kivymd-icon-256.png" + ), + text="Single-line item with avatar", + secondary_text: "Secondary text here", + ) + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/TwoLineRightIconListItem.png + :align: center + +.. ThreeLineRightIconListItem: +ThreeLineRightIconListItem +-------------------------- + +.. tabs:: + + .. tab:: Declarative KV style + + .. code-block:: kv + + ThreeLineRightIconListItem: + text: "Single-line item with avatar" + secondary_text: "Secondary text here" + tertiary_text: "fit more text than usual" + + ImageRightWidget: + source: "kivymd/images/logo/kivymd-icon-256.png" + + .. tab:: Declarative python style + + .. code-block:: python + + ThreeLineRightIconListItem( + ImageRightWidget( + source="kivymd/images/logo/kivymd-icon-256.png" + ), + text="Single-line item with avatar", + secondary_text: "Secondary text here", + tertiary_text: "fit more text than usual", + ) + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/ThreeLineRightIconListItem.png + :align: center + .. OneLineIconListItem: OneLineIconListItem ------------------- -.. code-block:: kv +.. tabs:: - OneLineIconListItem: - text: "Single-line item with avatar" + .. tab:: Declarative KV style - IconLeftWidget: - icon: "language-python" + .. code-block:: kv + + OneLineIconListItem: + text: "Single-line item with avatar" + + IconLeftWidget: + icon: "language-python" + + .. tab:: Declarative python style + + .. code-block:: python + + OneLineIconListItem( + IconLeftWidget( + icon="language-python" + ), + text="Single-line item with avatar" + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/OneLineIconListItem.png :align: center @@ -243,14 +477,30 @@ OneLineIconListItem TwoLineIconListItem ------------------- -.. code-block:: kv +.. tabs:: - TwoLineIconListItem: - text: "Two-line item with avatar" - secondary_text: "Secondary text here" + .. tab:: Declarative KV style - IconLeftWidget: - icon: "language-python" + .. code-block:: kv + + TwoLineIconListItem: + text: "Two-line item with avatar" + secondary_text: "Secondary text here" + + IconLeftWidget: + icon: "language-python" + + .. tab:: Declarative python style + + .. code-block:: python + + TwoLineIconListItem( + IconLeftWidget( + icon="language-python" + ), + text="Single-line item with avatar", + secondary_text: "Secondary text here" + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/TwoLineIconListItem.png :align: center @@ -259,15 +509,32 @@ TwoLineIconListItem ThreeLineIconListItem --------------------- -.. code-block:: kv +.. tabs:: - ThreeLineIconListItem: - text: "Three-line item with avatar" - secondary_text: "Secondary text here" - tertiary_text: "fit more text than usual" + .. tab:: Declarative KV style - IconLeftWidget: - icon: "language-python" + .. code-block:: kv + + ThreeLineIconListItem: + text: "Three-line item with avatar" + secondary_text: "Secondary text here" + tertiary_text: "fit more text than usual" + + IconLeftWidget: + icon: "language-python" + + .. tab:: Declarative python style + + .. code-block:: python + + ThreeLineIconListItem( + IconLeftWidget( + icon="language-python" + ), + text="Single-line item with avatar", + secondary_text: "Secondary text here", + tertiary_text: "fit more text than usual", + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/ThreeLineIconListItem.png :align: center @@ -276,16 +543,34 @@ ThreeLineIconListItem OneLineAvatarIconListItem ------------------------- -.. code-block:: kv +.. tabs:: - OneLineAvatarIconListItem: - text: "One-line item with avatar" + .. tab:: Declarative KV style - IconLeftWidget: - icon: "plus" + .. code-block:: kv - IconRightWidget: - icon: "minus" + OneLineAvatarIconListItem: + text: "One-line item with avatar" + + IconLeftWidget: + icon: "plus" + + IconRightWidget: + icon: "minus" + + .. tab:: Declarative python style + + .. code-block:: python + + OneLineAvatarIconListItem( + IconLeftWidget( + icon="plus" + ), + IconRightWidget( + icon="minus" + ), + text="Single-line item with avatar", + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/OneLineAvatarIconListItem.png :align: center @@ -294,17 +579,36 @@ OneLineAvatarIconListItem TwoLineAvatarIconListItem ------------------------- -.. code-block:: kv +.. tabs:: - TwoLineAvatarIconListItem: - text: "Two-line item with avatar" - secondary_text: "Secondary text here" + .. tab:: Declarative KV style - IconLeftWidget: - icon: "plus" + .. code-block:: kv - IconRightWidget: - icon: "minus" + TwoLineAvatarIconListItem: + text: "Two-line item with avatar" + secondary_text: "Secondary text here" + + IconLeftWidget: + icon: "plus" + + IconRightWidget: + icon: "minus" + + .. tab:: Declarative python style + + .. code-block:: python + + TwoLineAvatarIconListItem( + IconLeftWidget( + icon="plus" + ), + IconRightWidget( + icon="minus" + ), + text="Single-line item with avatar", + secondary_text: "Secondary text here", + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/TwoLineAvatarIconListItem.png :align: center @@ -313,18 +617,38 @@ TwoLineAvatarIconListItem ThreeLineAvatarIconListItem --------------------------- -.. code-block:: kv +.. tabs:: - ThreeLineAvatarIconListItem: - text: "Three-line item with avatar" - secondary_text: "Secondary text here" - tertiary_text: "fit more text than usual" + .. tab:: Declarative KV style - IconLeftWidget: - icon: "plus" + .. code-block:: kv - IconRightWidget: - icon: "minus" + ThreeLineAvatarIconListItem: + text: "Three-line item with avatar" + secondary_text: "Secondary text here" + tertiary_text: "fit more text than usual" + + IconLeftWidget: + icon: "plus" + + IconRightWidget: + icon: "minus" + + .. tab:: Declarative python style + + .. code-block:: python + + ThreeLineAvatarIconListItem( + IconLeftWidget( + icon="plus" + ), + IconRightWidget( + icon="minus" + ), + text="Single-line item with avatar", + secondary_text: "Secondary text here", + tertiary_text: "fit more text than usual", + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/ThreeLineAvatarIconListItem.png :align: center @@ -332,101 +656,196 @@ ThreeLineAvatarIconListItem Custom list item ---------------- -.. code-block:: python +.. tabs:: - from kivy.lang import Builder - from kivy.properties import StringProperty + .. tab:: Declarative KV style - from kivymd.app import MDApp - from kivymd.uix.list import IRightBodyTouch, OneLineAvatarIconListItem - from kivymd.uix.selectioncontrol import MDCheckbox - from kivymd.icon_definitions import md_icons + .. code-block:: python + + from kivy.lang import Builder + from kivy.properties import StringProperty + + from kivymd.app import MDApp + from kivymd.uix.list import IRightBodyTouch, OneLineAvatarIconListItem + from kivymd.uix.selectioncontrol import MDCheckbox + from kivymd.icon_definitions import md_icons - KV = ''' - : + KV = ''' + : - IconLeftWidget: - icon: root.icon + IconLeftWidget: + icon: root.icon - RightCheckbox: + RightCheckbox: - MDBoxLayout: + MDScrollView: - ScrollView: - - MDList: - id: scroll - ''' + MDList: + id: scroll + ''' - class ListItemWithCheckbox(OneLineAvatarIconListItem): - '''Custom list item.''' + class ListItemWithCheckbox(OneLineAvatarIconListItem): + '''Custom list item.''' - icon = StringProperty("android") + icon = StringProperty("android") - class RightCheckbox(IRightBodyTouch, MDCheckbox): - '''Custom right container.''' + class RightCheckbox(IRightBodyTouch, MDCheckbox): + '''Custom right container.''' - class MainApp(MDApp): - def build(self): - return Builder.load_string(KV) + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + return Builder.load_string(KV) - def on_start(self): - icons = list(md_icons.keys()) - for i in range(30): - self.root.ids.scroll.add_widget( - ListItemWithCheckbox(text=f"Item {i}", icon=icons[i]) - ) + def on_start(self): + icons = list(md_icons.keys()) + for i in range(30): + self.root.ids.scroll.add_widget( + ListItemWithCheckbox(text=f"Item {i}", icon=icons[i]) + ) - MainApp().run() + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.list import IRightBodyTouch, OneLineAvatarIconListItem + from kivymd.uix.selectioncontrol import MDCheckbox + from kivymd.uix.scrollview import MDScrollView + from kivymd.uix.list import MDList + from kivymd.icon_definitions import md_icons + + + class RightCheckbox(IRightBodyTouch, MDCheckbox): + '''Custom right container.''' + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + return ( + MDScrollView( + MDList( + id="scroll" + ) + ) + ) + + def on_start(self): + icons = list(md_icons.keys()) + for i in range(30): + self.root.ids.scroll.add_widget( + OneLineAvatarIconListItem( + IconLeftWidget( + icon=icons[i] + ), + RightCheckbox(), + text=f"Item {i}", + ) + ) + + + Example().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/custom-list-item.png :align: center -.. code-block:: python +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp - from kivymd.uix.boxlayout import MDBoxLayout - from kivymd.uix.list import IRightBodyTouch + .. code-block:: python - KV = ''' - OneLineAvatarIconListItem: - text: "One-line item with avatar" - on_size: - self.ids._right_container.width = container.width - self.ids._right_container.x = container.width + from kivy.lang import Builder - IconLeftWidget: - icon: "cog" + from kivymd.app import MDApp + from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.list import IRightBodyTouch - YourContainer: - id: container + KV = ''' + OneLineAvatarIconListItem: + text: "One-line item with avatar" + on_size: + self.ids._right_container.width = container.width + self.ids._right_container.x = container.width - MDIconButton: - icon: "minus" + IconLeftWidget: + icon: "cog" - MDIconButton: - icon: "plus" - ''' + YourContainer: + id: container + + MDIconButton: + icon: "minus" + + MDIconButton: + icon: "plus" + ''' - class YourContainer(IRightBodyTouch, MDBoxLayout): - adaptive_width = True + class YourContainer(IRightBodyTouch, MDBoxLayout): + adaptive_width = True - class MainApp(MDApp): - def build(self): - return Builder.load_string(KV) + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + return Builder.load_string(KV) - MainApp().run() + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.list import IRightBodyTouch + from kivymd.uix.button import MDIconButton + from kivymd.uix.list import OneLineAvatarIconListItem, IconLeftWidget + + + class YourContainer(IRightBodyTouch, MDBoxLayout): + adaptive_width = True + + + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + return ( + OneLineAvatarIconListItem( + IconLeftWidget( + icon="cog" + ), + YourContainer( + MDIconButton( + icon="minus" + ), + MDIconButton( + icon="plus" + ), + id="container" + ), + text="One-line item with avatar" + ) + ) + + def on_start(self): + container = self.root.ids.container + self.root.ids._right_container.width = container.width + container.x = container.width + + + Example().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/custom-list-right-container.png :align: center @@ -437,26 +856,56 @@ Behavior When using the `AvatarListItem` and `IconListItem` classes, when an icon is clicked, the event of this icon is triggered: -.. code-block:: kv +.. tabs:: - OneLineIconListItem: - text: "Single-line item with icon" + .. tab:: Declarative KV style - IconLeftWidget: - icon: "language-python" + .. code-block:: kv + + OneLineIconListItem: + text: "Single-line item with icon" + + IconLeftWidget: + icon: "language-python" + + .. tab:: Declarative python style + + .. code-block:: python + + OneLineIconListItem( + IconLeftWidget( + icon="language-python" + ), + text="Single-line item with avatar", + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/list-icon-trigger.gif :align: center You can disable the icon event using the `WithoutTouch` classes: -.. code-block:: kv +.. tabs:: - OneLineIconListItem: - text: "Single-line item with icon" + .. tab:: Declarative KV style - IconLeftWidgetWithoutTouch: - icon: "language-python" + .. code-block:: kv + + OneLineIconListItem: + text: "Single-line item with icon" + + IconLeftWidgetWithoutTouch: + icon: "language-python" + + .. tab:: Declarative python style + + .. code-block:: python + + OneLineIconListItem( + IconLeftWidgetWithoutTouch( + icon="language-python" + ), + text="Single-line item with avatar", + ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/list-icon-without-trigger.gif :align: center @@ -539,13 +988,9 @@ class MDList(MDGridLayout): _list_vertical_padding = NumericProperty("8dp") - def add_widget(self, widget, index=0, canvas=None): - super().add_widget(widget, index, canvas) - self.height += widget.height - - def remove_widget(self, widget): - super().remove_widget(widget) - self.height -= widget.height + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.adaptive_height = True class BaseListItem( @@ -569,8 +1014,8 @@ class BaseListItem( text_color = ColorProperty(None) """ - Text color in ``rgba`` format used if :attr:`~theme_text_color` is set - to `'Custom'`. + Text color in (r, g, b, a) or string format used + if :attr:`~theme_text_color` is set to `'Custom'`. :attr:`text_color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. @@ -578,7 +1023,9 @@ class BaseListItem( font_style = StringProperty("Subtitle1") """ - Text font style. See ``kivymd.font_definitions.py``. + Text font style. + See `font-definitions `_ + for more information. :attr:`font_style` is a :class:`~kivy.properties.StringProperty` and defaults to `'Subtitle1'`. @@ -586,7 +1033,7 @@ class BaseListItem( theme_text_color = StringProperty("Primary", allownone=True) """ - Theme text color in ``rgba`` format for primary text. + The name of the color scheme for for the primary text. :attr:`theme_text_color` is a :class:`~kivy.properties.StringProperty` and defaults to `'Primary'`. @@ -610,7 +1057,7 @@ class BaseListItem( secondary_text_color = ColorProperty(None) """ - Text color in ``rgba`` format used for secondary text + Text color in (r, g, b, a) or string format used for secondary text if :attr:`~secondary_theme_text_color` is set to `'Custom'`. :attr:`secondary_text_color` is a :class:`~kivy.properties.ColorProperty` @@ -619,7 +1066,7 @@ class BaseListItem( tertiary_text_color = ColorProperty(None) """ - Text color in ``rgba`` format used for tertiary text + Text color in (r, g, b, a) or string format used for tertiary text if :attr:`~tertiary_theme_text_color` is set to 'Custom'. :attr:`tertiary_text_color` is a :class:`~kivy.properties.ColorProperty` @@ -628,7 +1075,7 @@ class BaseListItem( secondary_theme_text_color = StringProperty("Secondary", allownone=True) """ - Theme text color for secondary text. + The name of the color scheme for for the secondary text. :attr:`secondary_theme_text_color` is a :class:`~kivy.properties.StringProperty` and defaults to `'Secondary'`. @@ -636,7 +1083,7 @@ class BaseListItem( tertiary_theme_text_color = StringProperty("Secondary", allownone=True) """ - Theme text color for tertiary text. + The name of the color scheme for for the tertiary text. :attr:`tertiary_theme_text_color` is a :class:`~kivy.properties.StringProperty` and defaults to `'Secondary'`. @@ -644,7 +1091,9 @@ class BaseListItem( secondary_font_style = StringProperty("Body1") """ - Font style for secondary line. See ``kivymd.font_definitions.py``. + Font style for secondary line. + See `font-definitions `_ + for more information. :attr:`secondary_font_style` is a :class:`~kivy.properties.StringProperty` and defaults to `'Body1'`. @@ -652,7 +1101,9 @@ class BaseListItem( tertiary_font_style = StringProperty("Body1") """ - Font style for tertiary line. See ``kivymd.font_definitions.py``. + Font style for tertiary line. + See `font-definitions `_ + for more information. :attr:`tertiary_font_style` is a :class:`~kivy.properties.StringProperty` and defaults to `'Body1'`. @@ -671,7 +1122,7 @@ class BaseListItem( divider_color = ColorProperty(None) """ - Divider color. + Divider color in (r, g, b, a) or string format. .. versionadded:: 1.0.0 @@ -681,7 +1132,7 @@ class BaseListItem( bg_color = ColorProperty(None) """ - Background color for menu item. + Background color for list item in (r, g, b, a) or string format. :attr:`bg_color` is a :class:`~kivy.properties.ColorProperty` and defaults to `None`. diff --git a/sbapp/kivymd/uix/menu/menu.kv b/sbapp/kivymd/uix/menu/menu.kv index d4ac30e..f72f87d 100644 --- a/sbapp/kivymd/uix/menu/menu.kv +++ b/sbapp/kivymd/uix/menu/menu.kv @@ -1,4 +1,4 @@ -#:import STD_INC kivymd.material_resources.STANDARD_INCREMENT +#:import STANDARD_INCREMENT kivymd.material_resources.STANDARD_INCREMENT @@ -14,7 +14,7 @@ size_hint: None, None - width: root.width_mult * STD_INC + width: root.width_mult * STANDARD_INCREMENT bar_width: 0 key_viewclass: "viewclass" key_size: "height" @@ -28,7 +28,7 @@ orientation: "vertical" - + diff --git a/sbapp/kivymd/uix/menu/menu.py b/sbapp/kivymd/uix/menu/menu.py index ebe1461..102cf96 100755 --- a/sbapp/kivymd/uix/menu/menu.py +++ b/sbapp/kivymd/uix/menu/menu.py @@ -781,7 +781,7 @@ class MDDropdownMenu(ThemableBehavior, FloatLayout): and defaults to `'[dp(7)]'`. """ - elevation = NumericProperty(10) + elevation = NumericProperty(4) """ Elevation value of menu dialog. @@ -790,7 +790,7 @@ class MDDropdownMenu(ThemableBehavior, FloatLayout): .. code-block:: python self.menu = MDDropdownMenu( - elevation=16, + elevation=4, ..., ) @@ -798,7 +798,7 @@ class MDDropdownMenu(ThemableBehavior, FloatLayout): :align: center :attr:`elevation` is an :class:`~kivy.properties.NumericProperty` - and defaults to `10`. + and defaults to `4`. """ _start_coords = [] diff --git a/sbapp/kivymd/uix/navigationdrawer/navigationdrawer.py b/sbapp/kivymd/uix/navigationdrawer/navigationdrawer.py index d315869..9ac5823 100755 --- a/sbapp/kivymd/uix/navigationdrawer/navigationdrawer.py +++ b/sbapp/kivymd/uix/navigationdrawer/navigationdrawer.py @@ -61,7 +61,7 @@ A simple example MDTopAppBar: title: "Navigation Drawer" - elevation: 10 + elevation: 4 pos_hint: {"top": 1} md_bg_color: "#e7e4c0" specific_text_color: "#4a4939" @@ -115,7 +115,7 @@ A simple example MDScreen( MDTopAppBar( title="Navigation Drawer", - elevation=10, + elevation=4, pos_hint={"top": 1}, md_bg_color="#e7e4c0", specific_text_color="#4a4939", @@ -188,7 +188,7 @@ Standard content for the navigation bar MDTopAppBar: title: "Navigation Drawer" - elevation: 10 + elevation: 4 pos_hint: {"top": 1} md_bg_color: "#e7e4c0" specific_text_color: "#4a4939" @@ -296,7 +296,7 @@ Standard content for the navigation bar MDScreen( MDTopAppBar( title="Navigation Drawer", - elevation=10, + elevation=4, pos_hint={"top": 1}, md_bg_color="#e7e4c0", specific_text_color="#4a4939", @@ -396,7 +396,7 @@ Switching screens in the ``ScreenManager`` and using the common ``MDTopAppBar`` MDTopAppBar: pos_hint: {"top": 1} - elevation: 10 + elevation: 4 title: "MDNavigationDrawer" left_action_items: [["menu", lambda x: nav_drawer.set_state("open")]] @@ -465,7 +465,7 @@ Switching screens in the ``ScreenManager`` and using the common ``MDTopAppBar`` MDScreen( MDTopAppBar( pos_hint={"top": 1}, - elevation=10, + elevation=4, title="MDNavigationDrawer", left_action_items=[["menu", lambda x: self.nav_drawer_open()]], ), @@ -551,14 +551,9 @@ from kivy.properties import ( StringProperty, VariableListProperty, ) -from kivy.uix.floatlayout import FloatLayout from kivy.uix.screenmanager import ScreenManager from kivymd import uix_path -from kivymd.uix.behaviors import ( - DeclarativeBehavior, - FakeRectangularElevationBehavior, -) from kivymd.uix.behaviors.focus_behavior import FocusBehavior from kivymd.uix.boxlayout import MDBoxLayout from kivymd.uix.card import MDCard @@ -1029,7 +1024,7 @@ class MDNavigationDrawerMenu(MDScrollView): widget.text_color = widget._text_color -class MDNavigationDrawer(MDCard, FakeRectangularElevationBehavior): +class MDNavigationDrawer(MDCard): type = OptionProperty("modal", options=("standard", "modal")) """ Type of drawer. Modal type will be on top of screen. Standard type will be diff --git a/sbapp/kivymd/uix/navigationrail/navigationrail.py b/sbapp/kivymd/uix/navigationrail/navigationrail.py index 8d0acdc..42e2e24 100644 --- a/sbapp/kivymd/uix/navigationrail/navigationrail.py +++ b/sbapp/kivymd/uix/navigationrail/navigationrail.py @@ -29,45 +29,86 @@ Usage MDNavigationRailItem: -.. code-block:: python +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp + .. code-block:: python - KV = ''' + from kivy.lang import Builder + + from kivymd.app import MDApp + + KV = ''' + MDBoxLayout: + + MDNavigationRail: + + MDNavigationRailItem: + text: "Python" + icon: "language-python" + + MDNavigationRailItem: + text: "JavaScript" + icon: "language-javascript" + + MDNavigationRailItem: + text: "CPP" + icon: "language-cpp" + + MDNavigationRailItem: + text: "Git" + icon: "git" + + MDScreen: + ''' - MDBoxLayout: - - MDNavigationRail: - - MDNavigationRailItem: - text: "Python" - icon: "language-python" - - MDNavigationRailItem: - text: "JavaScript" - icon: "language-javascript" - - MDNavigationRailItem: - text: "CPP" - icon: "language-cpp" - - MDNavigationRailItem: - text: "Git" - icon: "git" - - MDScreen: - ''' + class Example(MDApp): + def build(self): + return Builder.load_string(KV) - class Example(MDApp): - def build(self): - return Builder.load_string(KV) + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.navigationrail import MDNavigationRail, MDNavigationRailItem - Example().run() + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDBoxLayout( + MDNavigationRail( + MDNavigationRailItem( + text="Python", + icon="language-python", + ), + MDNavigationRailItem( + text="JavaScript", + icon="language-javascript", + ), + MDNavigationRailItem( + text="CPP", + icon="language-cpp", + ), + MDNavigationRailItem( + text="Git", + icon="git", + ), + ) + ) + ) + + + Example().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/navigation-rail-usage.png :align: center @@ -75,202 +116,412 @@ Usage Example ======= -.. code-block:: python +.. tabs:: - from kivy.clock import Clock - from kivy.lang import Builder + .. tab:: Declarative KV and imperative python styles - from kivymd.app import MDApp - from kivymd.uix.behaviors import RoundedRectangularElevationBehavior - from kivymd.uix.boxlayout import MDBoxLayout - from kivymd.uix.button import MDFillRoundFlatIconButton - from kivymd.uix.label import MDLabel - from kivymd.uix.screen import MDScreen + .. code-block:: python - KV = ''' - #:import FadeTransition kivy.uix.screenmanager.FadeTransition + from kivy.clock import Clock + from kivy.lang import Builder + + from kivymd.app import MDApp + from kivymd.uix.behaviors import CommonElevationBehavior + from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.button import MDFillRoundFlatIconButton + from kivymd.uix.label import MDLabel + from kivymd.uix.screen import MDScreen + + KV = ''' + #:import FadeTransition kivy.uix.screenmanager.FadeTransition - - elevation: 3 - -height: "56dp" + + elevation: 3.5 + shadow_radius: 12 + shadow_softness: 4 + -height: "56dp" - - focus_color: "#e7e4c0" - unfocus_color: "#fffcf4" + + focus_color: "#e7e4c0" + unfocus_color: "#fffcf4" - MDScreen: + MDScreen: - MDNavigationLayout: + MDNavigationLayout: - ScreenManager: + ScreenManager: - MDScreen: + MDScreen: - MDBoxLayout: - orientation: "vertical" + MDBoxLayout: + orientation: "vertical" + + MDBoxLayout: + adaptive_height: True + md_bg_color: "#fffcf4" + padding: "12dp" + + MDLabel: + text: "12:00" + adaptive_height: True + pos_hint: {"center_y": .5} + + MDBoxLayout: + + MDNavigationRail: + id: navigation_rail + md_bg_color: "#fffcf4" + selected_color_background: "#e7e4c0" + ripple_color_item: "#e7e4c0" + on_item_release: app.switch_screen(*args) + + MDNavigationRailMenuButton: + on_release: nav_drawer.set_state("open") + + MDNavigationRailFabButton: + md_bg_color: "#b0f0d6" + + MDNavigationRailItem: + text: "Python" + icon: "language-python" + + MDNavigationRailItem: + text: "JavaScript" + icon: "language-javascript" + + MDNavigationRailItem: + text: "CPP" + icon: "language-cpp" + + MDNavigationRailItem: + text: "Swift" + icon: "language-swift" + + ScreenManager: + id: screen_manager + transition: + FadeTransition(duration=.2, clearcolor=app.theme_cls.bg_dark) + + MDNavigationDrawer: + id: nav_drawer + radius: (0, 16, 16, 0) + md_bg_color: "#fffcf4" + elevation: 4 + width: "240dp" + + MDNavigationDrawerMenu: MDBoxLayout: + orientation: "vertical" adaptive_height: True - md_bg_color: "#fffcf4" - padding: "12dp" + spacing: "12dp" + padding: "3dp", 0, 0, "12dp" - MDLabel: - text: "12:00" - adaptive_height: True - pos_hint: {"center_y": .5} + MDIconButton: + icon: "menu" - MDBoxLayout: + ExtendedButton: + text: "Compose" + icon: "pencil" - MDNavigationRail: - id: navigation_rail - md_bg_color: "#fffcf4" - selected_color_background: "#e7e4c0" - ripple_color_item: "#e7e4c0" - on_item_release: app.switch_screen(*args) + DrawerClickableItem: + text: "Python" + icon: "language-python" - MDNavigationRailMenuButton: - on_release: nav_drawer.set_state("open") + DrawerClickableItem: + text: "JavaScript" + icon: "language-javascript" - MDNavigationRailFabButton: - md_bg_color: "#b0f0d6" + DrawerClickableItem: + text: "CPP" + icon: "language-cpp" - MDNavigationRailItem: - text: "Python" - icon: "language-python" - - MDNavigationRailItem: - text: "JavaScript" - icon: "language-javascript" - - MDNavigationRailItem: - text: "CPP" - icon: "language-cpp" - - MDNavigationRailItem: - text: "Swift" - icon: "language-swift" - - ScreenManager: - id: screen_manager - transition: - FadeTransition(duration=.2, clearcolor=app.theme_cls.bg_dark) - - MDNavigationDrawer: - id: nav_drawer - radius: (0, 16, 16, 0) - md_bg_color: "#fffcf4" - elevation: 12 - width: "240dp" - - MDNavigationDrawerMenu: - - MDBoxLayout: - orientation: "vertical" - adaptive_height: True - spacing: "12dp" - padding: 0, 0, 0, "12dp" - - MDIconButton: - icon: "menu" - - ExtendedButton: - text: "Compose" - icon: "pencil" - - DrawerClickableItem: - text: "Python" - icon: "language-python" - - DrawerClickableItem: - text: "JavaScript" - icon: "language-javascript" - - DrawerClickableItem: - text: "CPP" - icon: "language-cpp" - - DrawerClickableItem: - text: "Swift" - icon: "language-swift" - ''' - - - class ExtendedButton( - RoundedRectangularElevationBehavior, MDFillRoundFlatIconButton - ): - ''' - Implements a button of type - `Extended FAB `_. - - .. rubric:: - Extended FABs help people take primary actions. - They're wider than FABs to accommodate a text label and larger target - area. - - This type of buttons is not yet implemented in the standard widget set - of the KivyMD library, so we will implement it ourselves in this class. - ''' - - def __init__(self, **kwargs): - super().__init__(**kwargs) - self.padding = "16dp" - Clock.schedule_once(self.set_spacing) - - def set_spacing(self, interval): - self.ids.box.spacing = "12dp" - - def set_radius(self, *args): - if self.rounded_button: - self._radius = self.radius = self.height / 4 - - - class Example(MDApp): - def build(self): - self.theme_cls.material_style = "M3" - self.theme_cls.primary_palette = "Orange" - return Builder.load_string(KV) - - def switch_screen( - self, instance_navigation_rail, instance_navigation_rail_item - ): - ''' - Called when tapping on rail menu items. Switches application screens. + DrawerClickableItem: + text: "Swift" + icon: "language-swift" ''' - self.root.ids.screen_manager.current = ( - instance_navigation_rail_item.icon.split("-")[1].lower() + + class ExtendedButton(MDFillRoundFlatIconButton, CommonElevationBehavior): + ''' + Implements a button of type + `Extended FAB `_. + + .. rubric:: + Extended FABs help people take primary actions. + They're wider than FABs to accommodate a text label and larger target + area. + + This type of buttons is not yet implemented in the standard widget set + of the KivyMD library, so we will implement it ourselves in this class. + ''' + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.padding = "16dp" + Clock.schedule_once(self.set_spacing) + + def set_spacing(self, interval): + self.ids.box.spacing = "12dp" + + def set_radius(self, *args): + if self.rounded_button: + self._radius = self.radius = self.height / 4 + + + class Example(MDApp): + def build(self): + self.theme_cls.material_style = "M3" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) + + def switch_screen( + self, instance_navigation_rail, instance_navigation_rail_item + ): + ''' + Called when tapping on rail menu items. Switches application screens. + ''' + + self.root.ids.screen_manager.current = ( + instance_navigation_rail_item.icon.split("-")[1].lower() + ) + + def on_start(self): + '''Creates application screens.''' + + navigation_rail_items = self.root.ids.navigation_rail.get_items()[:] + navigation_rail_items.reverse() + + for widget in navigation_rail_items: + name_screen = widget.icon.split("-")[1].lower() + screen = MDScreen( + name=name_screen, + md_bg_color="#edd769", + radius=[18, 0, 0, 0], + ) + box = MDBoxLayout(padding="12dp") + label = MDLabel( + text=name_screen.capitalize(), + font_style="H1", + halign="right", + adaptive_height=True, + shorten=True, + ) + box.add_widget(label) + screen.add_widget(box) + self.root.ids.screen_manager.add_widget(screen) + + + Example().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivy.clock import Clock + from kivy.metrics import dp + + from kivymd.app import MDApp + from kivymd.uix.behaviors import CommonElevationBehavior + from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.button import MDFillRoundFlatIconButton, MDIconButton + from kivymd.uix.label import MDLabel + from kivymd.uix.navigationdrawer import ( + MDNavigationDrawerItem, + MDNavigationLayout, + MDNavigationDrawer, + MDNavigationDrawerMenu, ) - - def on_start(self): - '''Creates application screens.''' - - navigation_rail_items = self.root.ids.navigation_rail.get_items()[:] - navigation_rail_items.reverse() - - for widget in navigation_rail_items: - name_screen = widget.icon.split("-")[1].lower() - screen = MDScreen( - name=name_screen, - md_bg_color="#edd769", - radius=[18, 0, 0, 0], - ) - box = MDBoxLayout(padding="12dp") - label = MDLabel( - text=name_screen.capitalize(), - font_style="H1", - halign="right", - adaptive_height=True, - shorten=True, - ) - box.add_widget(label) - screen.add_widget(box) - self.root.ids.screen_manager.add_widget(screen) + from kivymd.uix.navigationrail import ( + MDNavigationRail, + MDNavigationRailMenuButton, + MDNavigationRailFabButton, + MDNavigationRailItem, + ) + from kivymd.uix.screen import MDScreen + from kivymd.uix.screenmanager import MDScreenManager - Example().run() + class DrawerClickableItem(MDNavigationDrawerItem): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.focus_color = "#e7e4c0" + self.unfocus_color = self.theme_cls.bg_light + self.radius = 24 + + + class ExtendedButton(MDFillRoundFlatIconButton, CommonElevationBehavior): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.padding = "16dp" + self.elevation = 3.5 + self.shadow_radius = 12 + self.shadow_softness = 4 + self.height = dp(56) + Clock.schedule_once(self.set_spacing) + + def set_spacing(self, interval): + self.ids.box.spacing = "12dp" + + def set_radius(self, *args): + if self.rounded_button: + self._radius = self.radius = self.height / 4 + + + class Example(MDApp): + def build(self): + self.theme_cls.material_style = "M3" + self.theme_cls.primary_palette = "Orange" + return MDScreen( + MDNavigationLayout( + MDScreenManager( + MDScreen( + MDBoxLayout( + MDBoxLayout( + MDLabel( + text="12:00", + adaptive_height=True, + pos_hint={"center_y": 0.5}, + ), + adaptive_height=True, + md_bg_color="#fffcf4", + padding="12dp", + ), + MDBoxLayout( + MDNavigationRail( + MDNavigationRailMenuButton( + on_release=self.open_nav_drawer, + ), + MDNavigationRailFabButton( + md_bg_color="#b0f0d6", + ), + MDNavigationRailItem( + text="Python", + icon="language-python", + ), + MDNavigationRailItem( + text="JavaScript", + icon="language-javascript", + ), + MDNavigationRailItem( + text="CPP", + icon="language-cpp", + ), + MDNavigationRailItem( + text="Swift", + icon="language-swift", + ), + id="navigation_rail", + md_bg_color="#fffcf4", + selected_color_background="#e7e4c0", + ripple_color_item="#e7e4c0", + ), + MDScreenManager( + id="screen_manager_content", + ), + id="root_box", + ), + id="box_rail", + orientation="vertical", + ), + id="box", + ), + id="screen", + ), + id="screen_manager", + ), + MDNavigationDrawer( + MDNavigationDrawerMenu( + MDBoxLayout( + MDIconButton( + icon="menu", + ), + ExtendedButton( + text="Compose", + icon="pencil", + ), + orientation="vertical", + adaptive_height=True, + spacing="12dp", + padding=("3dp", 0, 0, "12dp"), + ), + DrawerClickableItem( + text="Python", + icon="language-python", + ), + DrawerClickableItem( + text="JavaScript", + icon="language-javascript", + ), + DrawerClickableItem( + text="CPP", + icon="language-cpp", + ), + DrawerClickableItem( + text="Swift", + icon="language-swift", + ), + ), + id="nav_drawer", + radius=(0, 16, 16, 0), + elevation=4, + width="240dp", + ), + ) + + def switch_screen(self, *args, screen_manager_content=None): + ''' + Called when tapping on rail menu items. Switches application screens. + ''' + + instance_navigation_rail, instance_navigation_rail_item = args + screen_manager_content.current = ( + instance_navigation_rail_item.icon.split("-")[1].lower() + ) + + def open_nav_drawer(self, *args): + self.root.ids.nav_drawer.set_state("open") + + def on_start(self): + '''Creates application screens.''' + + screen_manager = self.root.ids.screen_manager + root_box = screen_manager.ids.screen.ids.box.ids.box_rail.ids.root_box + navigation_rail = root_box.ids.navigation_rail + screen_manager_content = root_box.ids.screen_manager_content + navigation_rail_items = navigation_rail.get_items()[:] + navigation_rail_items.reverse() + navigation_rail.bind( + on_item_release=lambda *args: self.switch_screen( + *args, screen_manager_content=screen_manager_content + ) + ) + + for widget in navigation_rail_items: + name_screen = widget.icon.split("-")[1].lower() + screen_manager_content.add_widget( + MDScreen( + MDBoxLayout( + MDLabel( + text=name_screen.capitalize(), + font_style="H1", + halign="right", + adaptive_height=True, + shorten=True, + ), + padding="12dp", + ), + name=name_screen, + md_bg_color="#edd769", + radius=[18, 0, 0, 0], + ), + ) + + + Example().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/navigation-rail-example.gif :align: center @@ -306,12 +557,11 @@ from kivy.uix.behaviors import ButtonBehavior from kivymd import uix_path from kivymd.theming import ThemableBehavior -from kivymd.uix.behaviors import FakeRectangularElevationBehavior +from kivymd.uix.behaviors import ScaleBehavior from kivymd.uix.boxlayout import MDBoxLayout from kivymd.uix.button import MDFloatingActionButton, MDIconButton from kivymd.uix.card import MDCard from kivymd.uix.floatlayout import MDFloatLayout -from kivymd.uix.templates import ScaleWidget from kivymd.uix.widget import MDWidget with open( @@ -333,7 +583,7 @@ class PanelItems(MDBoxLayout): """Box for menu items.""" -class RippleWidget(MDWidget, ScaleWidget): +class RippleWidget(MDWidget, ScaleBehavior): """ Implements a background color for a menu item - (:class:`~MDNavigationRailItem`). @@ -562,7 +812,7 @@ class MDNavigationRailItem(ThemableBehavior, ButtonBehavior, MDBoxLayout): self.navigation_rail.dispatch("on_item_release", self) -class MDNavigationRail(MDCard, FakeRectangularElevationBehavior): +class MDNavigationRail(MDCard): """ :Events: :attr:`on_item_press` diff --git a/sbapp/kivymd/uix/pickers/datepicker/datepicker.kv b/sbapp/kivymd/uix/pickers/datepicker/datepicker.kv index 8f95e58..13835de 100644 --- a/sbapp/kivymd/uix/pickers/datepicker/datepicker.kv +++ b/sbapp/kivymd/uix/pickers/datepicker/datepicker.kv @@ -307,34 +307,6 @@ else (dp(32), dp(32)) disabled: True - canvas: - Color: - rgba: - ( \ - ( \ - self.theme_cls.primary_color if not root.owner.selector_color \ - else root.owner.selector_color \ - ) \ - if root.is_selected and not self.disabled \ - else (0, 0, 0, 0) \ - ) \ - if self.owner.mode != "range" else \ - ( \ - ( \ - self.theme_cls.primary_color if not root.owner.selector_color \ - else root.owner.selector_color \ - ) \ - if root.is_selected and not self.disabled \ - and (self.owner.mode == "range" and self.owner._start_range_date) \ - else (0, 0, 0, 0) \ - ) - Ellipse: - size: - (dp(42), dp(42)) \ - if root.theme_cls.device_orientation == "portrait" \ - else (dp(32), dp(32)) - pos: self.pos - # Fill marking the available dates of the range, if using the `range` mode # or use `min_date/max_date`. canvas.before: @@ -355,12 +327,14 @@ if root.theme_cls.device_orientation == "portrait" \ else \ (dp(32), dp(28)) \ - if self.index in [6, 13, 20, 27, 30] or self.owner._date_range \ + if self.index in [6, 13, 20, 27, 34] or self.owner._date_range \ and self.text and self.owner._date_range[-1] == date( \ self.current_year, \ self.current_month, \ int(self.text) \ ) \ + or self.text and int(self.text) == \ + calendar.monthrange(self.current_year, self.current_month)[1] \ else (dp(46), dp(28)) pos: (self.x - dp(1.5), self.y + dp(5)) \ @@ -395,29 +369,15 @@ else [0, 0, 0, 0]) \ ) - # Circle marking the beginning and end of the date range if the "range" - # mode is used. + # Selection circle. Color: rgba: - [0, 0, 0, 0] if not self.owner._date_range else \ - ( ( \ self.theme_cls.primary_color if not root.owner.selector_color \ else root.owner.selector_color \ ) \ - if self.text and self.owner._date_range[0] == date( \ - self.current_year, \ - self.current_month, \ - int(self.text) \ - ) \ - or \ - self.text and self.owner._date_range[-1] == date( \ - self.current_year, \ - self.current_month, \ - int(self.text) \ - ) \ - else (0, 0, 0, 0) \ - ) + if root.is_selected and not self.disabled \ + else (0, 0, 0, 0) Ellipse: size: (dp(42), dp(42)) \ diff --git a/sbapp/kivymd/uix/pickers/datepicker/datepicker.py b/sbapp/kivymd/uix/pickers/datepicker/datepicker.py index 50e80de..3188408 100644 --- a/sbapp/kivymd/uix/pickers/datepicker/datepicker.py +++ b/sbapp/kivymd/uix/pickers/datepicker/datepicker.py @@ -11,65 +11,110 @@ Components/DatePicker .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/picker-previous.png :align: center -.. warning:: The widget is under testing. Therefore, we would be grateful if - you would let us know about the bugs found. - .. rubric:: Usage -.. code-block:: python +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp - from kivymd.uix.pickers import MDDatePicker + .. code-block:: python - KV = ''' - MDFloatLayout: + from kivy.lang import Builder - MDTopAppBar: - title: "MDDatePicker" - pos_hint: {"top": 1} - elevation: 10 + from kivymd.app import MDApp + from kivymd.uix.pickers import MDDatePicker - MDRaisedButton: - text: "Open date picker" - pos_hint: {'center_x': .5, 'center_y': .5} - on_release: app.show_date_picker() - ''' + KV = ''' + MDFloatLayout: - - class Test(MDApp): - def build(self): - return Builder.load_string(KV) - - def on_save(self, instance, value, date_range): - ''' - Events called when the "OK" dialog box button is clicked. - - :type instance: ; - - :param value: selected date; - :type value: ; - - :param date_range: list of 'datetime.date' objects in the selected range; - :type date_range: ; + MDRaisedButton: + text: "Open date picker" + pos_hint: {'center_x': .5, 'center_y': .5} + on_release: app.show_date_picker() ''' - print(instance, value, date_range) - def on_cancel(self, instance, value): - '''Events called when the "CANCEL" dialog box button is clicked.''' + class Test(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) - def show_date_picker(self): - date_dialog = MDDatePicker() - date_dialog.bind(on_save=self.on_save, on_cancel=self.on_cancel) - date_dialog.open() + def on_save(self, instance, value, date_range): + ''' + Events called when the "OK" dialog box button is clicked. + + :type instance: ; + :param value: selected date; + :type value: ; + :param date_range: list of 'datetime.date' objects in the selected range; + :type date_range: ; + ''' + + print(instance, value, date_range) + + def on_cancel(self, instance, value): + '''Events called when the "CANCEL" dialog box button is clicked.''' + + def show_date_picker(self): + date_dialog = MDDatePicker() + date_dialog.bind(on_save=self.on_save, on_cancel=self.on_cancel) + date_dialog.open() - Test().run() + Test().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.button import MDRaisedButton + from kivymd.uix.pickers import MDDatePicker + from kivymd.uix.screen import MDScreen -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDDatePicker.gif + class Test(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDRaisedButton( + text="Open data picker", + pos_hint={'center_x': .5, 'center_y': .5}, + on_release=self.show_date_picker, + ) + ) + ) + + def on_save(self, instance, value, date_range): + ''' + Events called when the "OK" dialog box button is clicked. + + :type instance: ; + + :param value: selected date; + :type value: ; + + :param date_range: list of 'datetime.date' objects in the selected range; + :type date_range: ; + ''' + + print(instance, value, date_range) + + def on_cancel(self, instance, value): + '''Events called when the "CANCEL" dialog box button is clicked.''' + + def show_date_picker(self, *args): + date_dialog = MDDatePicker() + date_dialog.bind(on_save=self.on_save, on_cancel=self.on_cancel) + date_dialog.open() + + + Test().run() + +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDDatePicker.png :align: center Open date dialog with the specified date @@ -81,7 +126,7 @@ Open date dialog with the specified date date_dialog = MDDatePicker(year=1983, month=4, day=12) date_dialog.open() -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/previous-date.png +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/specified-date.png :align: center Interval date @@ -94,12 +139,16 @@ that are not included in this range will have the status `disabled`. def show_date_picker(self): date_dialog = MDDatePicker( - min_date=datetime.date(2021, 2, 15), - max_date=datetime.date(2021, 3, 27), + min_date=datetime.date.today(), + max_date=datetime.date( + datetime.date.today().year, + datetime.date.today().month, + datetime.date.today().day + 2, + ), ) date_dialog.open() -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/range-date.gif +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/range-date.png :align: center The range of available dates can be changed in the picker dialog: @@ -122,7 +171,7 @@ You can set the range of years using the :attr:`~kivymd.uix.picker.MDDatePicker. .. code-block:: python def show_date_picker(self): - date_dialog = MDDatePicker(min_year=2021, max_year=2030) + date_dialog = MDDatePicker(min_year=2022, max_year=2030) date_dialog.open() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/min-max-year-date.png @@ -141,18 +190,21 @@ Set and select a date range :align: center """ +from __future__ import annotations + __all__ = ("MDDatePicker", "BaseDialogPicker", "DatePickerInputField") import calendar import datetime +import math import os import time from datetime import date +from itertools import zip_longest from typing import Union from kivy import Logger from kivy.animation import Animation -from kivy.clock import Clock from kivy.lang import Builder from kivy.metrics import dp from kivy.properties import ( @@ -175,7 +227,7 @@ from kivymd.theming import ThemableBehavior, ThemeManager from kivymd.toast import toast from kivymd.uix.behaviors import ( CircularRippleBehavior, - FakeRectangularElevationBehavior, + CommonElevationBehavior, SpecificBackgroundColorBehavior, ) from kivymd.uix.boxlayout import MDBoxLayout @@ -194,7 +246,7 @@ with open( class BaseDialogPicker( BaseDialog, - FakeRectangularElevationBehavior, + CommonElevationBehavior, SpecificBackgroundColorBehavior, ): """ @@ -255,11 +307,11 @@ class BaseDialogPicker( primary_color = ColorProperty(None) """ - Background color of toolbar in (r, g, b, a) format. + Background color of toolbar in (r, g, b, a) or string format. .. code-block:: python - MDDatePicker(primary_color=get_color_from_hex("#72225b")) + MDDatePicker(primary_color="brown") .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/primary-color-date.png :align: center @@ -270,13 +322,13 @@ class BaseDialogPicker( accent_color = ColorProperty(None) """ - Background color of calendar/clock face in (r, g, b, a) format. + Background color of calendar/clock face in (r, g, b, a) or string format. .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), + primary_color="brown", + accent_color="darkred", ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/accent-color-date.png @@ -288,14 +340,15 @@ class BaseDialogPicker( selector_color = ColorProperty(None) """ - Background color of the selected day of the month or hour in (r, g, b, a) format. + Background color of the selected day of the month or hour in (r, g, b, a) + or string format. .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - selector_color=get_color_from_hex("#e93f39"), + primary_color="brown", + accent_color="darkred", + selector_color="red", ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/selector-color-date.png @@ -307,15 +360,15 @@ class BaseDialogPicker( text_toolbar_color = ColorProperty(None) """ - Color of labels for text on a toolbar in (r, g, b, a) format. + Color of labels for text on a toolbar in (r, g, b, a) or string format. .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - selector_color=get_color_from_hex("#e93f39"), - text_toolbar_color=get_color_from_hex("#cccccc"), + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-toolbar-color-date.png @@ -327,16 +380,16 @@ class BaseDialogPicker( text_color = ColorProperty(None) """ - Color of text labels in calendar/clock face in (r, g, b, a) format. + Color of text labels in calendar/clock face in (r, g, b, a) or string format. .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - selector_color=get_color_from_hex("#e93f39"), - text_toolbar_color=get_color_from_hex("#cccccc"), - text_color=("#ffffff"), + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", + text_color="orange", ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-color-date.png @@ -348,17 +401,18 @@ class BaseDialogPicker( text_current_color = ColorProperty(None) """ - Color of the text of the current day of the month/hour in (r, g, b, a) format. + Color of the text of the current day of the month/hour in (r, g, b, a) + or string format. .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - selector_color=get_color_from_hex("#e93f39"), - text_toolbar_color=get_color_from_hex("#cccccc"), - text_color=("#ffffff"), - text_current_color=get_color_from_hex("#e93f39"), + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", + text_color="orange", + text_current_color="white", ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-current-color-date.png @@ -375,13 +429,13 @@ class BaseDialogPicker( .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - selector_color=get_color_from_hex("#e93f39"), - text_toolbar_color=get_color_from_hex("#cccccc"), - text_color=("#ffffff"), - text_current_color=get_color_from_hex("#e93f39"), - text_button_color=(1, 1, 1, .5), + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", + text_color="orange", + text_current_color="white", + text_button_color="lightgrey", ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-button-color-date.png @@ -391,52 +445,124 @@ class BaseDialogPicker( and defaults to `None`. """ - input_field_background_color = ColorProperty(None) + input_field_background_color_normal = ColorProperty(None) """ - Background color of input fields in (r, g, b, a) format. + Background color normal of input fields in (r, g, b, a) or string format. + + .. versionadded:: 1.1.0 .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - selector_color=get_color_from_hex("#e93f39"), - text_toolbar_color=get_color_from_hex("#cccccc"), - text_color=("#ffffff"), - text_current_color=get_color_from_hex("#e93f39"), - input_field_background_color=(1, 1, 1, 0.2), + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", + text_color="orange", + text_current_color="white", + text_button_color="lightgrey", + input_field_background_color_normal="coral", ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/input-field-background-color-date.png :align: center - :attr:`input_field_background_color` is an :class:`~kivy.properties.ColorProperty` + :attr:`input_field_background_color_normal` is an :class:`~kivy.properties.ColorProperty` and defaults to `None`. """ + input_field_background_color_focus = ColorProperty(None) + """ + Background color normal of input fields in (r, g, b, a) or string format. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + MDDatePicker( + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", + text_color="orange", + text_current_color="white", + text_button_color="lightgrey", + input_field_background_color_normal="coral", + input_field_background_color_focus="red", + ) + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/input-field-background-color-focus-date.png + :align: center + + :attr:`input_field_background_color_focus` is an :class:`~kivy.properties.ColorProperty` + and defaults to `None`. + """ + + input_field_background_color = ColorProperty(None) + """ + .. deprecated:: 1.1.0 + Use :attr:`input_field_background_color_normal` instead. + """ + input_field_text_color = ColorProperty(None) """ - Text color of input fields in (r, g, b, a) format. + .. deprecated:: 1.1.0 + Use :attr:`input_field_text_color_normal` instead. + """ - Background color of input fields. + input_field_text_color_normal = ColorProperty(None) + """ + Text color normal of input fields in (r, g, b, a) or string format. + + .. versionadded:: 1.1.0 .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - selector_color=get_color_from_hex("#e93f39"), - text_toolbar_color=get_color_from_hex("#cccccc"), - text_color=("#ffffff"), - text_current_color=get_color_from_hex("#e93f39"), - input_field_background_color=(1, 1, 1, 0.2), - input_field_text_color=(1, 1, 1, 1), + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", + text_color="orange", + text_current_color="white", + text_button_color="lightgrey", + input_field_background_color_normal="brown", + input_field_background_color_focus="red", + input_field_text_color_normal="white", ) - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/input-field-background-color-date.png + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/input-field-text-color-normal-date.png :align: center - :attr:`input_field_text_color` is an :class:`~kivy.properties.ColorProperty` + :attr:`input_field_text_color_normal` is an :class:`~kivy.properties.ColorProperty` + and defaults to `None`. + """ + + input_field_text_color_focus = ColorProperty(None) + """ + Text color focus of input fields in (r, g, b, a) or string format. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + MDDatePicker( + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", + text_color="orange", + text_current_color="white", + text_button_color="lightgrey", + input_field_background_color_normal="brown", + input_field_background_color_focus="red", + input_field_text_color_normal="white", + ) + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/input-field-text-color-normal-date.png + :align: center + + :attr:`input_field_text_color_focus` is an :class:`~kivy.properties.ColorProperty` and defaults to `None`. """ @@ -447,16 +573,18 @@ class BaseDialogPicker( .. code-block:: python MDDatePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - selector_color=get_color_from_hex("#e93f39"), - text_toolbar_color=get_color_from_hex("#cccccc"), - text_color=("#ffffff"), - text_current_color=get_color_from_hex("#e93f39"), - input_field_background_color=(1, 1, 1, 0.2), - input_field_text_color=(1, 1, 1, 1), - font_name="Weather.ttf", - + primary_color="brown", + accent_color="darkred", + selector_color="red", + text_toolbar_color="lightgrey", + text_color="orange", + text_current_color="white", + text_button_color="lightgrey", + input_field_background_color_normal="brown", + input_field_background_color_focus="red", + input_field_text_color_normal="white", + input_field_text_color_focus="lightgrey", + font_name="nasalization.ttf", ) .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/font-name-date.png @@ -471,6 +599,20 @@ class BaseDialogPicker( self.register_event_type("on_save") self.register_event_type("on_cancel") + def on_input_field_background_color( + self, instance, value: str | list | tuple + ) -> None: + """For supported of current API.""" + + self.input_field_background_color_normal = value + + def on_input_field_text_color( + self, instance, value: str | list | tuple + ) -> None: + """For supported of current API.""" + + self.input_field_text_color_normal = value + def on_save(self, *args) -> None: """Events called when the "OK" dialog box button is clicked.""" @@ -606,13 +748,18 @@ class DatePickerDaySelectableItem( self.owner.set_selected_widget(self) + def on_touch_down(self, touch): + # If year_layout is active don't dispatch on_touch_down events, + # so date items don't consume touch. + if not self.owner.ids._year_layout.disabled: + return + super().on_touch_down(touch) + class DatePickerYearSelectableItem(RecycleDataViewBehavior, MDLabel): """Implements an item for a pick list of the year.""" index = None - selected = BooleanProperty(False) - selectable = BooleanProperty(True) selected_color = ColorProperty([0, 0, 0, 0]) owner = ObjectProperty() @@ -623,7 +770,7 @@ class DatePickerYearSelectableItem(RecycleDataViewBehavior, MDLabel): def on_touch_down(self, touch): if super().on_touch_down(touch): return True - if self.collide_point(*touch.pos) and self.selectable: + if self.collide_point(*touch.pos): self.owner.year = int(self.text) # self.owner.sel_year = self.owner.year self.owner.ids.label_full_date.text = self.owner.set_text_full_date( @@ -635,7 +782,6 @@ class DatePickerYearSelectableItem(RecycleDataViewBehavior, MDLabel): return self.parent.select_with_touch(self.index, touch) def apply_selection(self, table_data, index, is_selected): - self.selected = is_selected if is_selected: self.selected_color = ( self.owner.selector_color @@ -661,7 +807,7 @@ class DatePickerYearSelectableItem(RecycleDataViewBehavior, MDLabel): class MDDatePicker(BaseDialogPicker): text_weekday_color = ColorProperty(None) """ - Text color of weekday names in (r, g, b, a) format. + Text color of weekday names in (r, g, b, a) or string format. .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-date-picker-text-weekday-color.png :align: center @@ -813,7 +959,6 @@ class MDDatePicker(BaseDialogPicker): _enter_data_field_two = None _enter_data_field_container = None _date_range = [] - _sel_day_widget = ObjectProperty() _scale_calendar_layout = NumericProperty(1) _scale_year_layout = NumericProperty(0) _shift_dialog_height = NumericProperty(0) @@ -838,11 +983,6 @@ class MDDatePicker(BaseDialogPicker): self.month = self.sel_month self.year = self.sel_year self.day = self.sel_day - self._current_selected_date = ( - self.sel_day, - self.sel_month, - self.sel_year, - ) super().__init__(**kwargs) self.theme_cls.bind(device_orientation=self.on_device_orientation) @@ -861,16 +1001,6 @@ class MDDatePicker(BaseDialogPicker): self.generate_list_widgets_days() self.update_calendar(self.sel_year, self.sel_month) - if ( - not self.max_date - and not self.min_date - and not self._date_range - and self.mode != "range" - ): - # Mark the current day. - self.set_month_day(self.sel_day) - self._sel_day_widget.dispatch("on_release") - def on_device_orientation( self, instance_theme_manager: ThemeManager, orientation_value: str ) -> None: @@ -924,18 +1054,14 @@ class MDDatePicker(BaseDialogPicker): Animation(opacity=1, d=0.15).start(self.ids.chevron_left) Animation(opacity=1, d=0.15).start(self.ids.chevron_right) Animation(_scale_year_layout=0, d=0.15).start(self) - Animation( - _shift_dialog_height=dp(0), _scale_calendar_layout=1, d=0.15 - ).start(self) + Animation(_scale_calendar_layout=1, d=0.15).start(self) - self._calendar_layout.clear_widgets() - self.generate_list_widgets_days() + # Move selection to the same day and month of the selected year. + self.sel_year = self.year + last_day = calendar.monthrange(self.year, self.sel_month)[1] + self.sel_day = min(self.sel_day, last_day) self.update_calendar(self.year, self.month) - if self.mode != "range": - self.set_month_day(self.day) - self._sel_day_widget.dispatch("on_release") - def transformation_to_dialog_select_year(self) -> None: def disabled_chevron_buttons(*args): self.ids.chevron_left.disabled = True @@ -943,15 +1069,20 @@ class MDDatePicker(BaseDialogPicker): self._select_year_dialog_open = True self.ids._year_layout.disabled = False - self._scale_calendar_layout = 0 Animation(opacity=0, d=0.15).start(self.ids.chevron_left) Animation(opacity=0, d=0.15).start(self.ids.chevron_right) + Animation(_scale_calendar_layout=0, d=0.15).start(self) anim = Animation(_scale_year_layout=1, d=0.15) anim.bind(on_complete=disabled_chevron_buttons) anim.start(self) self.ids.triangle.icon = "menu-up" self.generate_list_widgets_years() self.set_position_to_current_year() + if self.min_year <= self.year < self.max_year: + index = self.year - self.min_year + self.ids._year_layout.children[0].select_node(index) + else: + self.ids._year_layout.children[0].clear_selection() def transformation_to_dialog_input_date(self) -> None: def set_date_to_input_field(): @@ -1063,13 +1194,10 @@ class MDDatePicker(BaseDialogPicker): if not self.min_date and not self.max_date: list_date = self._enter_data_field.get_list_date() if len(list_date) == 3 and len(list_date[2]) == 4: - # self._sel_day_widget.is_selected = False - self.update_calendar(int(list_date[2]), int(list_date[1])) - self.set_month_day(int(list_date[0])) - # self._sel_day_widget.dispatch("on_release") - if self.mode != "range": - self._sel_day_widget.is_selected = False - self._sel_day_widget.dispatch("on_release") + self.sel_day = int(list_date[0]) + self.sel_month = int(list_date[1]) + self.sel_year = int(list_date[2]) + self.update_calendar(self.sel_year, self.sel_month) elif self.min_date and self.max_date: list_min_date = self._enter_data_field.get_list_date() list_max_date = self._enter_data_field_two.get_list_date() @@ -1107,8 +1235,6 @@ class MDDatePicker(BaseDialogPicker): def update_calendar_for_date_range(self) -> None: # self.compare_date_range() self._date_range = self.get_date_range() - self._calendar_layout.clear_widgets() - self.generate_list_widgets_days() self.update_calendar(self.year, self.month) def update_text_full_date(self, list_date) -> None: @@ -1140,79 +1266,72 @@ class MDDatePicker(BaseDialogPicker): ) def update_calendar(self, year, month) -> None: - try: - dates = [x for x in self.calendar.itermonthdates(year, month)] - except ValueError as e: - if str(e) == "year is out of range": - pass + self.year, self.month = year, month + if self.mode == "picker": + selected_date = date(self.sel_year, self.sel_month, self.sel_day) + selected_dates = {selected_date} else: - self.year = year - self.month = month - for idx in range(len(self._calendar_list)): - self._calendar_list[idx].current_month = int(self.month) - self._calendar_list[idx].current_year = int(self.year) - - # Dates of the month not in the range 1-31. - if idx >= len(dates) or dates[idx].month != month: - # self._calendar_list[idx].disabled = True - self._calendar_list[idx].text = "" - # Dates of the month in the range 1-31. - else: - self._calendar_list[idx].disabled = False - self._calendar_list[idx].text = str(dates[idx].day) - self._calendar_list[idx].is_today = dates[idx] == self.today - # The marked date widget has a True value in the `is_selected` - # attribute. In the KV file it is checked if the date widget - # (DatePickerDaySelectableItem) has the `is_selected = False` - # attribute value, then the date widget is not highlighted. - if ( - 0 - if not self._calendar_list[idx].text - else int(self._calendar_list[idx].text), - self._calendar_list[idx].current_month, - self._calendar_list[idx].current_year, - ) == self._current_selected_date: - self._calendar_list[idx].is_selected = True - else: - self._calendar_list[idx].is_selected = False - # Dates outside the set range - disabled. - if ( - self.mode == "picker" - and self._date_range - and self._calendar_list[idx].text - ) or ( - self.mode == "range" - and self._start_range_date - and self._end_range_date - and self._calendar_list[idx].text - ): - if ( - date( - self._calendar_list[idx].current_year, - self._calendar_list[idx].current_month, - int(self._calendar_list[idx].text), - ) - not in self._date_range - ): - self._calendar_list[idx].disabled = True + selected_dates = {self._start_range_date, self._end_range_date} + dates = self.calendar.itermonthdates(year, month) + for widget, widget_date in zip_longest(self._calendar_list, dates): + # Only widgets whose dates are in the displayed month are visible. + visible = ( + widget_date is not None + and widget_date.month == month + and widget_date.year == year + ) + widget.text = str(widget_date.day) if visible else "" + widget.current_year = year + widget.current_month = month + widget.is_today = visible and widget_date == self.today + widget.is_selected = visible and widget_date in selected_dates + # I don't understand why, but this line is important. Without this + # line, some widgets that we are trying to disable remain enabled. + widget.disabled = False + widget.disabled = ( + not visible + or self.mode == "range" + and self._date_range + and widget_date not in self._date_range + ) def get_field(self) -> MDTextField: """Creates and returns a text field object used to enter dates.""" if issubclass(self.input_field_cls, MDTextField): + text_color_focus = ( + self.input_field_text_color_focus + if self.input_field_text_color_focus + else self.theme_cls.primary_color + ) + text_color_normal = ( + self.input_field_text_color_normal + if self.input_field_text_color_normal + else self.theme_cls.disabled_hint_text_color + ) + fill_color_focus = ( + self.input_field_background_color_focus + if self.input_field_background_color_focus + else self.theme_cls.bg_dark + ) + fill_color_normal = ( + self.input_field_background_color_normal + if self.input_field_background_color_normal + else self.theme_cls.bg_darkest + ) + field = self.input_field_cls( owner=self, helper_text=self.helper_text, - line_color_normal=self.theme_cls.divider_color, + fill_color_normal=fill_color_normal, + fill_color_focus=fill_color_focus, + hint_text_color_normal=text_color_normal, + hint_text_color_focus=text_color_focus, + text_color_normal=text_color_normal, + text_color_focus=text_color_focus, + line_color_focus=text_color_focus, + line_color_normal=text_color_normal, ) - field.color_mode = "custom" - field.line_color_focus = ( - self.theme_cls.primary_color - if not self.input_field_text_color - else self.input_field_text_color - ) - field.current_hint_text_color = field.line_color_focus - field._current_hint_text_color = field.line_color_focus return field else: raise TypeError( @@ -1239,10 +1358,7 @@ class MDDatePicker(BaseDialogPicker): "set_text_full_date:\n\t" f"Month [{month}] out of range." ) if int(day) > calendar.monthrange(int(year), (month))[1]: - raise ValueError( - "set_text_full_date:\n\t" - f"Day [{day}] out of range for the month {month}" - ) + return "" date = datetime.date(int(year), int(month), int(day)) separator = ( "\n" @@ -1345,46 +1461,55 @@ class MDDatePicker(BaseDialogPicker): ) def set_selected_widget(self, widget) -> None: - if self._sel_day_widget: - self._sel_day_widget.is_selected = False - - widget.is_selected = True - self.sel_month = int(self.month) - self.sel_year = int(self.year) + self.sel_year = self.year + self.sel_month = self.month self.sel_day = int(widget.text) - self._current_selected_date = ( - self.sel_day, - self.sel_month, - self.sel_year, - ) - self._sel_day_widget = widget + self.update_calendar(self.sel_year, self.sel_month) def set_month_day(self, day) -> None: - for idx in range(len(self._calendar_list)): - if str(day) == str(self._calendar_list[idx].text): - self._sel_day_widget = self._calendar_list[idx] - self.sel_day = int(self._calendar_list[idx].text) - if self._sel_day_widget: - self._sel_day_widget.is_selected = False - self._sel_day_widget = self._calendar_list[idx] + # This method is no longer used. The code bellow repeats the behavior + # that was previously required of it for backward compatibility + # reasons. + self.sel_day = day + self.update_calendar(self.sel_year, self.sel_month) def set_position_to_current_year(self) -> None: - # TODO: Add the feature to set the position of the list of years - # for the current year. This is not currently possible because the - # ``RecycleView`` class does not support this functionality. - # There is a solution to this problem - # - https://github.com/Bakterija/log_fruit/blob/dev/src/app_modules/widgets/app_recycleview/recycleview.py. - # But I have not been able to get it to work. - pass + year_layout = self.ids._year_layout + # When this method is called for the first time, RecycleView has not + # yet added widgets to the year list, so we use the default height. + widget_height = year_layout.children[0].default_size[1] + cols_amount = year_layout.children[0].cols + rows_amount = math.ceil((self.max_year - self.min_year) / cols_amount) + row_index = (self.year - self.min_year) // cols_amount + # To find the middle of the current year widget, we add the height of + # the rows under this widget with half the widget height. + widget_center_y = (rows_amount - row_index - 1 + 0.5) * widget_height + viewport_height = year_layout.height + year_list_height = rows_amount * widget_height + # If there are too few years in the list to fill the entire viewport, + # RecycleView displays additional empty space outside the list. + # We have to move the viewport up so that this space is displayed + # under the years list. Also, this guard condition protects against + # the division by zero error below. + if viewport_height >= year_list_height: + year_layout.scroll_y = 1 + return + viewport_bottom = widget_center_y - 0.5 * viewport_height + # We set scroll_y property to the ratio of the actual lifting height + # of the viewport to the maximum possible, and clamp this ratio in the + # range from 0 to 1 so that the viewport still is in a valid position + # if it is impossible to show the widget in the middle. + scroll_y = viewport_bottom / (year_list_height - viewport_height) + year_layout.scroll_y = min(1, max(0, scroll_y)) def generate_list_widgets_years(self) -> None: + self.ids._year_layout.data = [] for i, number_year in enumerate(range(self.min_year, self.max_year)): self.ids._year_layout.data.append( { "owner": self, "text": str(number_year), "index": i, - "selectable": True, "viewclass": "DatePickerYearSelectableItem", } ) @@ -1416,26 +1541,9 @@ class MDDatePicker(BaseDialogPicker): Called when "chevron-left" and "chevron-right" buttons are pressed. Switches the calendar to the previous/next month. """ - - operation = 1 if operation == "next" else -1 - month = ( - 12 - if self.month + operation == 0 - else 1 - if self.month + operation == 13 - else self.month + operation - ) - year = ( - self.year - 1 - if self.month + operation == 0 - else self.year + 1 - if self.month + operation == 13 - else self.year - ) + month_delta = 1 if operation == "next" else -1 + year = self.year + (self.month - 1 + month_delta) // 12 + month = (self.month - 1 + month_delta) % 12 + 1 + if year <= 0: + year, month = 1, 1 self.update_calendar(year, month) - if self.sel_day: - x = calendar.monthrange(year, month)[1] - if x < self.sel_day: - self.sel_day = ( - x if year <= self.sel_year and month <= self.sel_year else 1 - ) diff --git a/sbapp/kivymd/uix/pickers/timepicker/timepicker.py b/sbapp/kivymd/uix/pickers/timepicker/timepicker.py index cb1e875..2b646d8 100644 --- a/sbapp/kivymd/uix/pickers/timepicker/timepicker.py +++ b/sbapp/kivymd/uix/pickers/timepicker/timepicker.py @@ -16,35 +16,73 @@ Components/TimePicker .. rubric:: Usage -.. code-block:: +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp - from kivymd.uix.pickers import MDTimePicker + .. code-block:: python - KV = ''' - MDFloatLayout: + from kivy.lang import Builder - MDRaisedButton: - text: "Open time picker" - pos_hint: {'center_x': .5, 'center_y': .5} - on_release: app.show_time_picker() - ''' + from kivymd.app import MDApp + from kivymd.uix.pickers import MDTimePicker + + KV = ''' + MDFloatLayout: + + MDRaisedButton: + text: "Open time picker" + pos_hint: {'center_x': .5, 'center_y': .5} + on_release: app.show_time_picker() + ''' - class Test(MDApp): - def build(self): - return Builder.load_string(KV) + class Test(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) - def show_time_picker(self): - '''Open time picker dialog.''' + def show_time_picker(self): + '''Open time picker dialog.''' - time_dialog = MDTimePicker() - time_dialog.open() + time_dialog = MDTimePicker() + time_dialog.open() - Test().run() + Test().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.button import MDRaisedButton + from kivymd.uix.pickers import MDTimePicker + from kivymd.uix.screen import MDScreen + + + class Test(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDRaisedButton( + text="Open time picker", + pos_hint={'center_x': .5, 'center_y': .5}, + on_release=self.show_time_picker, + ) + ) + ) + + def show_time_picker(self, *args): + '''Open time picker dialog.''' + + MDTimePicker().open() + + + Test().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/MDTimePicker.png :align: center @@ -91,14 +129,14 @@ Use the :attr:`~MDTimePicker.set_time` method of the .. code-block:: python - time_dialog = MDTimePicker( - primary_color=get_color_from_hex("#72225b"), - accent_color=get_color_from_hex("#5d1a4a"), - text_button_color=(1, 1, 1, 1), - ) + MDTimePicker( + primary_color="brown", + accent_color="red", + text_button_color="white", + ).open() - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/time-picker-customization.png - :align: center +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/time-picker-customization.png + :align: center """ __all__ = ("MDTimePicker",) @@ -194,8 +232,8 @@ class TimeInputTextField(MDTextField): hour_regx = "^[0-9]$|^0[1-9]$|^1[0-2]$" minute_regx = "^[0-9]$|^0[0-9]$|^[1-5][0-9]$" - def __init__(self, **kwargs): - super().__init__(**kwargs) + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) Clock.schedule_once(self.set_text) self.register_event_type("on_select") self.bind(text_color_focus=self.setter("hint_text_color_normal")) @@ -217,17 +255,20 @@ class TimeInputTextField(MDTextField): to somehow make them aligned. """ - if not self.text: - self.text = " " + def set_text(*args): + if not self.text: + self.text = " " - self._refresh_text(self.text) - max_size = max(self._lines_rects, key=lambda r: r.size[0]).size - dx = (self.width - max_size[0]) / 2.0 - dy = (self.height - max_size[1]) / 2.0 - self.padding = [dx, dy, dx, dy] + self._refresh_text(self.text) + max_size = max(self._lines_rects, key=lambda r: r.size[0]).size + dx = (self.width - max_size[0]) / 2.0 + dy = (self.height - max_size[1]) / 2.0 + self.padding = [dx, dy, dx, dy] - if len(self.text) > 1: - self.text = self.text.replace(" ", "") + if len(self.text) > 1: + self.text = self.text.replace(" ", "") + + Clock.schedule_once(set_text) def on_focus(self, *args) -> None: super().on_focus(*args) diff --git a/sbapp/kivymd/uix/refreshlayout/refreshlayout.py b/sbapp/kivymd/uix/refreshlayout/refreshlayout.py index e6aafad..170b76e 100755 --- a/sbapp/kivymd/uix/refreshlayout/refreshlayout.py +++ b/sbapp/kivymd/uix/refreshlayout/refreshlayout.py @@ -36,7 +36,7 @@ Example title: app.title md_bg_color: app.theme_cls.primary_color background_palette: 'Primary' - elevation: 10 + elevation: 4 left_action_items: [['menu', lambda x: x]] MDScrollViewRefreshLayout: diff --git a/sbapp/kivymd/uix/screen.py b/sbapp/kivymd/uix/screen.py index 609e507..2e399ec 100644 --- a/sbapp/kivymd/uix/screen.py +++ b/sbapp/kivymd/uix/screen.py @@ -29,7 +29,7 @@ MDScreen md_bg_color: app.theme_cls.primary_color """ -from kivy.properties import ObjectProperty +from kivy.properties import ListProperty, ObjectProperty from kivy.uix.screenmanager import Screen from kivymd.uix import MDAdaptiveWidget @@ -44,20 +44,34 @@ class MDScreen(DeclarativeBehavior, Screen, MDAdaptiveWidget): see in the :class:`~kivy.uix.screenmanager.Screen` class documentation. """ - hero_to = ObjectProperty() + hero_to = ObjectProperty(deprecated=True) """ - Must be a :class:`~kivymd.uix.hero.MDHeroTo` class. + Must be a :class:`~kivymd.uix.hero.MDHeroTo` class. + See the documentation of the `MDHeroTo `_ widget for more detailed information. - .. versionchanged:: 1.0.0 + .. deprecated:: 1.0.0 + Use attr:`heroes_to` attribute instead. :attr:`hero_to` is an :class:`~kivy.properties.ObjectProperty` and defaults to `None`. """ - def on_hero_to(self, screen, widget) -> None: + heroes_to = ListProperty() + """ + Must be a list of :class:`~kivymd.uix.hero.MDHeroTo` class. + + .. versionadded:: 1.0.0 + + :attr:`heroes_to` is an :class:`~kivy.properties.LiatProperty` + and defaults to `[]`. + """ + + def on_hero_to(self, screen, widget: MDHeroTo) -> None: + """Called when the value of the :attr:`hero_to` attribute changes.""" + if not isinstance(widget, MDHeroTo) or not issubclass( widget.__class__, MDHeroTo ): @@ -65,3 +79,4 @@ class MDScreen(DeclarativeBehavior, Screen, MDAdaptiveWidget): f"The `{widget}` widget must be an `kivymd.uix.hero.MDHeroTo` " f"class or inherited from this class" ) + self.heroes_to = [widget] diff --git a/sbapp/kivymd/uix/screenmanager.py b/sbapp/kivymd/uix/screenmanager.py index 6bf9554..dcc4792 100644 --- a/sbapp/kivymd/uix/screenmanager.py +++ b/sbapp/kivymd/uix/screenmanager.py @@ -8,8 +8,23 @@ Components/ScreenManager If you want to use Hero animations you need to use :class:`~kivymd.uix.screenmanager.MDScreenManager` not :class:`~kivy.uix.screenmanager.ScreenManager` class. + +Transition +---------- + +:class:`~kivymd.uix.screenmanager.MDScreenManager` class supports the following +transitions: + +- :class:`~kivymd.uix.transition.MDFadeSlideTransition` +- :class:`~kivymd.uix.transition.MDSlideTransition` +- :class:`~kivymd.uix.transition.MDSwapTransition` + +You need to use the :class:`~kivymd.uix.screenmanager.MDScreenManager` class +when you want to use hero animations on your screens. If you don't need hero +animation use the :class:`~kivy.uix.screenmanager.ScreenManager` class. """ +from kivy import Logger from kivy.clock import Clock from kivy.properties import ListProperty, StringProperty from kivy.uix.screenmanager import ScreenManager @@ -21,17 +36,22 @@ from kivymd.uix.hero import MDHeroFrom class MDScreenManager(DeclarativeBehavior, ScreenManager): """ Screen manager. This is the main class that will control your - :class:`~kivymd.uix.screen.MDScreen` stack and memory. For more + :class:`~kivymd.uix.screen.MDScreen` stack and memory. + + For more information, see in the :class:`~kivy.uix.screenmanager.ScreenManager` class documentation. """ - current_hero = StringProperty(None) + current_hero = StringProperty(None, deprecated=True) """ The name of the current tag for the :class:`~kivymd.uix.hero.MDHeroFrom` and :class:`~kivymd.uix.hero.MDHeroTo` objects that will be animated when animating the transition between screens. + .. deprecated:: 1.1.0 + Use :attr:`current_heroes` attribute instead. + See the `Hero `_ module documentation for more information about creating and using Hero animations. @@ -40,6 +60,17 @@ class MDScreenManager(DeclarativeBehavior, ScreenManager): and defaults to `None`. """ + current_heroes = ListProperty() + """ + A list of names (tags) of heroes that need to be animated when moving + to the next screen. + + .. versionadded:: 1.1.0 + + :attr:`current_heroes` is an :class:`~kivy.properties.ListProperty` + and defaults to `[]`. + """ + # Collection of `MDHeroFrom` objects on all screens of the current # screen manager. _heroes_data = ListProperty() @@ -58,28 +89,48 @@ class MDScreenManager(DeclarativeBehavior, ScreenManager): self.transition = MDSlideTransition() - def get_hero_from_widget(self) -> None: + def get_hero_from_widget(self) -> list: """ - Get an :class:`~kivymd.uix.hero.MDHeroTo` object with the - :attr:`~current_hero` tag. + Get a list of :class:`~kivymd.uix.hero.MDHeroFrom` objects according + to the tag names specified in the :attr:`~current_heroes` list. """ - hero_from_widget = None + hero_from_widget = [] - for hero_widget in self._heroes_data: - if isinstance(hero_widget, MDHeroFrom) or issubclass( - hero_widget.__class__, MDHeroFrom - ): - if hero_widget.tag == self.current_hero: - hero_from_widget = hero_widget - break + for name_hero in self.current_heroes: + for hero_widget in self._heroes_data: + if isinstance(hero_widget, MDHeroFrom) or issubclass( + hero_widget.__class__, MDHeroFrom + ): + if hero_widget.tag == name_hero: + hero_from_widget.append(hero_widget) return hero_from_widget + def on_current_hero(self, instance, value: str) -> None: + """ + Called when the value of the :attr:`current_hero` attribute changes. + """ + + Logger.warning( + "KivyMD: " + "`kivymd/uix/screenmanager.MDScreenManager.current_hero` " + "attribute is deprecated. " + "Use `kivymd/uix/screenmanager.MDScreenManager.current_heroes` " + "attribute instead." + ) + if value: + self.current_heroes = [value] + else: + self.current_heroes = [] + def add_widget(self, widget, *args, **kwargs): super().add_widget(widget, *args, **kwargs) Clock.schedule_once(lambda x: self._create_heroes_data(widget)) + # TODO: Add a method to delete an object from the arrt:`_heroes_data` + # collection when deleting an object using the `remove_widget` method. + def _create_heroes_data(self, widget): def find_hero_widget(child_widget): widget_hero = None diff --git a/sbapp/kivymd/uix/segmentedcontrol/segmentedcontrol.kv b/sbapp/kivymd/uix/segmentedcontrol/segmentedcontrol.kv index 9aed57f..3ac3656 100644 --- a/sbapp/kivymd/uix/segmentedcontrol/segmentedcontrol.kv +++ b/sbapp/kivymd/uix/segmentedcontrol/segmentedcontrol.kv @@ -15,8 +15,11 @@ pos_hint: {"center_y": .5} x: root._segment_switch_x md_bg_color: root.segment_color - elevation: 6 + elevation: 2 _radius: root.radius[0] - 4 + width: + segment_panel.width / segment_panel.children_number \ + - segment_panel.spacing SegmentPanel: id: segment_panel diff --git a/sbapp/kivymd/uix/segmentedcontrol/segmentedcontrol.py b/sbapp/kivymd/uix/segmentedcontrol/segmentedcontrol.py index 6a518c2..5e9c7f4 100644 --- a/sbapp/kivymd/uix/segmentedcontrol/segmentedcontrol.py +++ b/sbapp/kivymd/uix/segmentedcontrol/segmentedcontrol.py @@ -10,58 +10,77 @@ Components/SegmentedControl Usage ===== -.. code-block:: python +.. tabs:: - from kivy.lang import Builder + .. tab:: Declarative KV style - from kivymd.app import MDApp + .. code-block:: python + + from kivy.lang import Builder + + from kivymd.app import MDApp - KV = ''' - MDScreen: + KV = ''' + MDScreen: - MDSegmentedControl: - pos_hint: {"center_x": .5, "center_y": .5} + MDSegmentedControl: + pos_hint: {"center_x": .5, "center_y": .5} - MDSegmentedControlItem: - text: "Male" + MDSegmentedControlItem: + text: "Male" - MDSegmentedControlItem: - text: "Female" + MDSegmentedControlItem: + text: "Female" - MDSegmentedControlItem: - text: "All" - ''' + MDSegmentedControlItem: + text: "All" + ''' - class Test(MDApp): - def build(self): - return Builder.load_string(KV) + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) - Test().run() + Example().run() -Or only in python code: + .. tab:: Declarative python style -.. code-block:: python + .. code-block:: python - from kivymd.app import MDApp - from kivymd.uix.screen import MDScreen - from kivymd.uix.segmentedcontrol import MDSegmentedControl, MDSegmentedControlItem + from kivymd.app import MDApp + from kivymd.uix.screen import MDScreen + from kivymd.uix.segmentedcontrol import ( + MDSegmentedControl, MDSegmentedControlItem + ) - class Test(MDApp): - def build(self): - screen = MDScreen() - segment_control = MDSegmentedControl(pos_hint={"center_x": .5, "center_y": .5}) - segment_control.add_widget(MDSegmentedControlItem(text="Male")) - segment_control.add_widget(MDSegmentedControlItem(text="Female")) - segment_control.add_widget(MDSegmentedControlItem(text="All")) - screen.add_widget(segment_control) - return screen + class Example(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDSegmentedControl( + MDSegmentedControlItem( + text="Male" + ), + MDSegmentedControlItem( + text="Female" + ), + MDSegmentedControlItem( + text="All" + ), + pos_hint={"center_x": 0.5, "center_y": 0.5} + ) + ) + ) - Test().run() + Example().run() .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-segmented-control-usage.gif :align: center @@ -117,12 +136,22 @@ with open( class MDSegmentedControlItem(MDLabel): - """Implements a label to place on the :class:`~SegmentPanel` panel.""" + """ + Implements a label to place on the :class:`~SegmentPanel` panel. + + See :class:`~kivymd.uix.label.MDLabel` class documentation for more + information. + """ # TODO: Add an attribute for the color of the active segment label. class MDSegmentedControl(MDRelativeLayout, ThemableBehavior): """ + Implements a segmented control panel. + + Relative layout class. For more information, see in the + :class:`~kivy.uix.relativelayout.RelativeLayout` class documentation. + :Events: `on_active` Called when the segment is activated. @@ -135,7 +164,7 @@ class MDSegmentedControl(MDRelativeLayout, ThemableBehavior): .. code-block:: kv MDSegmentedControl: - md_bg_color: "#451938" + md_bg_color: "brown" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-segmented-control-md-bg-color.png :align: center @@ -151,8 +180,8 @@ class MDSegmentedControl(MDRelativeLayout, ThemableBehavior): .. code-block:: kv MDSegmentedControl: - md_bg_color: "#451938" - segment_color: "#e4514f" + md_bg_color: "brown" + segment_color: "red" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-segmented-control-segment-color.png :align: center @@ -160,8 +189,8 @@ class MDSegmentedControl(MDRelativeLayout, ThemableBehavior): .. code-block:: kv MDSegmentedControl: - md_bg_color: "#451938" - segment_color: "#e4514f" + md_bg_color: "brown" + segment_color: "red" MDSegmentedControlItem: text: "[color=fff]Male[/color]" @@ -196,9 +225,9 @@ class MDSegmentedControl(MDRelativeLayout, ThemableBehavior): .. code-block:: kv MDSegmentedControl: - md_bg_color: "#451938" - segment_color: "#e4514f" - separator_color: 1, 1, 1, 1 + md_bg_color: "brown" + segment_color: "red" + separator_color: "white" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/md-segmented-control-separator-color.png :align: center @@ -255,9 +284,6 @@ class MDSegmentedControl(MDRelativeLayout, ThemableBehavior): Clock.schedule_once(self.set_default_colors) Clock.schedule_once(self._remove_last_separator) - # FIXME: Sometimes this interval is not enough to get the width - # of the segment label textures. - Clock.schedule_once(self._set_width_segment_switch, 2.2) def set_default_colors(self, *args) -> None: """ @@ -313,6 +339,10 @@ class MDSegmentedControl(MDRelativeLayout, ThemableBehavior): self.ids.segment_panel.add_widget(widget) separator = MDSeparator(orientation="vertical") self.ids.segment_panel.add_widget(separator) + if not self.ids.segment_panel._started: + self.ids.segment_panel._started = True + else: + self.ids.segment_panel.children_number += 1 Clock.schedule_once( lambda x: self.update_separator_color(separator) ) @@ -326,15 +356,6 @@ class MDSegmentedControl(MDRelativeLayout, ThemableBehavior): self.current_active_segment = widget self.dispatch("on_active", widget) - def _set_width_segment_switch(self, *args): - """ - Sets the width of the switch. I think this is not done quite correctly. - """ - - self.ids.segment_switch.width = self.ids.segment_panel.children[ - 0 - ].width + dp(12) - def _remove_last_separator(self, *args): self.ids.segment_panel.remove_widget(self.ids.segment_panel.children[0]) @@ -350,3 +371,7 @@ class SegmentPanel(MDBoxLayout): Implements a panel for placing items - :class:`~MDSegmentedControlItem` for the :class:`~MDSegmentedControl` class. """ + + children_number = NumericProperty(1) + + _started = BooleanProperty(defaultvalue=False) diff --git a/sbapp/kivymd/uix/selectioncontrol/selectioncontrol.kv b/sbapp/kivymd/uix/selectioncontrol/selectioncontrol.kv index 1e7480a..3d3b015 100644 --- a/sbapp/kivymd/uix/selectioncontrol/selectioncontrol.kv +++ b/sbapp/kivymd/uix/selectioncontrol/selectioncontrol.kv @@ -101,7 +101,7 @@ size_hint: None, None size: dp(24), dp(24) elevation: - (8 if root.active else 5) \ + (2.5 if root.active else 1) \ if app.theme_cls.material_style != "M3" else \ 0 pos: diff --git a/sbapp/kivymd/uix/selectioncontrol/selectioncontrol.py b/sbapp/kivymd/uix/selectioncontrol/selectioncontrol.py index 48d0c9b..031fd3a 100755 --- a/sbapp/kivymd/uix/selectioncontrol/selectioncontrol.py +++ b/sbapp/kivymd/uix/selectioncontrol/selectioncontrol.py @@ -192,15 +192,10 @@ from kivy.properties import ( ) from kivy.uix.behaviors import ToggleButtonBehavior from kivy.uix.floatlayout import FloatLayout -from kivy.utils import get_color_from_hex from kivymd import uix_path -from kivymd.color_definitions import colors from kivymd.theming import ThemableBehavior -from kivymd.uix.behaviors import ( - CircularRippleBehavior, - FakeCircularElevationBehavior, -) +from kivymd.uix.behaviors import CircularRippleBehavior, CommonElevationBehavior from kivymd.uix.floatlayout import MDFloatLayout from kivymd.uix.label import MDIcon @@ -361,8 +356,10 @@ class MDCheckbox(CircularRippleBehavior, ToggleButtonBehavior, MDIcon): disabled=self.update_color, state=self.update_color, ) - self.theme_cls.bind(primary_color=self.update_primary_color) - self.theme_cls.bind(theme_style=self.update_primary_color) + self.theme_cls.bind( + theme_style=self.update_primary_color, + primary_color=self.update_primary_color, + ) self.update_icon() self.update_color() @@ -423,7 +420,7 @@ class ThumbIcon(MDIcon): class Thumb( - FakeCircularElevationBehavior, + CommonElevationBehavior, CircularRippleBehavior, MDFloatLayout, ): diff --git a/sbapp/kivymd/uix/slider/slider.kv b/sbapp/kivymd/uix/slider/slider.kv index 4f36344..5332381 100644 --- a/sbapp/kivymd/uix/slider/slider.kv +++ b/sbapp/kivymd/uix/slider/slider.kv @@ -3,7 +3,7 @@ #:import colors kivymd.color_definitions.colors - + @@ -131,16 +131,20 @@ ) \ ) \ ) - elevation: 0 if root._is_off else (4 if root.active else 2) + elevation: 0 if root._is_off else (3 if root.active else 1) HintBoxContainer: id: hint_box size_hint: None, None - md_bg_color: root.hint_bg_color - elevation: 0 + md_bg_color: root.hint_bg_color if root.hint_bg_color else [0, 0, 0, 0] + elevation: 1.5 opacity: 1 if root.active else 0 radius: root.hint_radius padding: "6dp", "6dp", "6dp", "8dp" + shadow_color: + ([0, 0, 0, 0.6] if root.hint_bg_color else [0, 0, 0, 0]) \ + if root.active else \ + [0, 0, 0, 0] size: lbl_value.width + self.padding[0] * 2, \ lbl_value.height + self.padding[0] diff --git a/sbapp/kivymd/uix/slider/slider.py b/sbapp/kivymd/uix/slider/slider.py index 5925abc..152a602 100644 --- a/sbapp/kivymd/uix/slider/slider.py +++ b/sbapp/kivymd/uix/slider/slider.py @@ -82,7 +82,7 @@ class MDSlider(ThemableBehavior, Slider): and defaults to `True`. """ - hint_bg_color = ColorProperty([0, 0, 0, 0]) + hint_bg_color = ColorProperty(None) """ Hint rectangle color in (r.g.b.a) format. diff --git a/sbapp/kivymd/uix/sliverappbar/sliverappbar.py b/sbapp/kivymd/uix/sliverappbar/sliverappbar.py index 2bfb242..6316315 100644 --- a/sbapp/kivymd/uix/sliverappbar/sliverappbar.py +++ b/sbapp/kivymd/uix/sliverappbar/sliverappbar.py @@ -38,8 +38,8 @@ Example from kivy.lang.builder import Builder + from kivymd.app import MDApp from kivymd.uix.card import MDCard - from kivymd.uix.behaviors import RoundedRectangularElevationBehavior KV = ''' @@ -47,7 +47,6 @@ Example height: "86dp" padding: "4dp" radius: 12 - elevation: 4 FitImage: source: "avatar.jpg" @@ -95,8 +94,10 @@ Example ''' - class CardItem(MDCard, RoundedRectangularElevationBehavior): - pass + class CardItem(MDCard): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.elevation = 3 class Example(MDApp): @@ -192,7 +193,6 @@ class MDSliverAppbar(MDBoxLayout, ThemableBehavior): from kivymd.uix.card import MDCard from kivymd.uix.toolbar import MDTopAppBar - from kivymd.uix.behaviors import RoundedRectangularElevationBehavior KV = ''' #:import SliverToolbar __main__.SliverToolbar @@ -203,7 +203,6 @@ class MDSliverAppbar(MDBoxLayout, ThemableBehavior): height: "86dp" padding: "4dp" radius: 12 - elevation: 4 FitImage: source: "avatar.jpg" @@ -252,13 +251,16 @@ class MDSliverAppbar(MDBoxLayout, ThemableBehavior): ''' - class CardItem(MDCard, RoundedRectangularElevationBehavior): - pass + class CardItem(MDCard): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.elevation = 3 class SliverToolbar(MDTopAppBar): def __init__(self, **kwargs): super().__init__(**kwargs) + self.shadow_color = (0, 0, 0, 0) self.type_height = "medium" self.headline_text = "Headline medium" self.left_action_items = [["arrow-left", lambda x: x]] @@ -422,6 +424,7 @@ class MDSliverAppbar(MDBoxLayout, ThemableBehavior): # Adding a custom MDTopAppBar object. if issubclass(instance_toolbar_cls.__class__, MDTopAppBar): instance_toolbar_cls.pos_hint = {"top": 1} + instance_toolbar_cls.elevation = 0 self.ids.float_box.add_widget(instance_toolbar_cls) else: raise MDSliverAppbarException( diff --git a/sbapp/kivymd/uix/snackbar/snackbar.kv b/sbapp/kivymd/uix/snackbar/snackbar.kv index 4dcef8b..557ceda 100644 --- a/sbapp/kivymd/uix/snackbar/snackbar.kv +++ b/sbapp/kivymd/uix/snackbar/snackbar.kv @@ -8,7 +8,7 @@ padding: "10dp", "10dp", "10dp", "10dp" md_bg_color: "323232" if not root.bg_color else root.bg_color radius: root.radius - elevation: 11 if root.padding else 0 + elevation: 4 if root.padding else 0 canvas: Color: diff --git a/sbapp/kivymd/uix/snackbar/snackbar.py b/sbapp/kivymd/uix/snackbar/snackbar.py index 6acb96c..c4fbdc9 100755 --- a/sbapp/kivymd/uix/snackbar/snackbar.py +++ b/sbapp/kivymd/uix/snackbar/snackbar.py @@ -284,7 +284,6 @@ from kivy.properties import ( ) from kivymd import uix_path -from kivymd.uix.behaviors import FakeRectangularElevationBehavior from kivymd.uix.button import BaseButton from kivymd.uix.card import MDCard @@ -294,7 +293,7 @@ with open( Builder.load_string(kv_file.read()) -class BaseSnackbar(MDCard, FakeRectangularElevationBehavior): +class BaseSnackbar(MDCard): """ :Events: :attr:`on_open` diff --git a/sbapp/kivymd/uix/swiper/swiper.py b/sbapp/kivymd/uix/swiper/swiper.py index bcf88d0..b89e2e5 100644 --- a/sbapp/kivymd/uix/swiper/swiper.py +++ b/sbapp/kivymd/uix/swiper/swiper.py @@ -38,7 +38,7 @@ Example MDTopAppBar: id: toolbar title: "MDSwiper" - elevation: 10 + elevation: 4 pos_hint: {"top": 1} MDSwiper: @@ -142,7 +142,7 @@ Example MDTopAppBar: id: toolbar title: "MDSwiper" - elevation: 10 + elevation: 4 pos_hint: {"top": 1} MDSwiper: @@ -203,7 +203,6 @@ from kivy.animation import Animation from kivy.clock import Clock from kivy.core.window import Window from kivy.effects.dampedscroll import DampedScrollEffect -from kivy.event import EventDispatcher from kivy.lang.builder import Builder from kivy.properties import ( BooleanProperty, @@ -212,7 +211,6 @@ from kivy.properties import ( StringProperty, ) from kivy.uix.anchorlayout import AnchorLayout -from kivy.uix.boxlayout import BoxLayout from kivy.utils import platform from kivymd import uix_path @@ -294,7 +292,7 @@ class MDSwiperItem(MDBoxLayout): anim.start(self) -class MDSwiper(MDScrollView, EventDispatcher): +class MDSwiper(MDScrollView): items_spacing = NumericProperty("20dp") """ The space between each :class:`MDSwiperItem`. diff --git a/sbapp/kivymd/uix/tab/tab.kv b/sbapp/kivymd/uix/tab/tab.kv index 824f82e..f219bc1 100644 --- a/sbapp/kivymd/uix/tab/tab.kv +++ b/sbapp/kivymd/uix/tab/tab.kv @@ -79,6 +79,10 @@ layout: layout size_hint: 1, None elevation: root.elevation + radius: root.radius + shadow_offset: root.shadow_offset + shadow_color: root.shadow_color + shadow_softness: root.shadow_softness height: root.tab_bar_height md_bg_color: self.theme_cls.primary_color \ diff --git a/sbapp/kivymd/uix/tab/tab.py b/sbapp/kivymd/uix/tab/tab.py index 6c93dbf..816711e 100755 --- a/sbapp/kivymd/uix/tab/tab.py +++ b/sbapp/kivymd/uix/tab/tab.py @@ -944,7 +944,6 @@ from kivy.properties import ( from kivy.uix.anchorlayout import AnchorLayout from kivy.uix.behaviors import ToggleButtonBehavior from kivy.uix.scrollview import ScrollView -from kivy.uix.widget import Widget from kivy.utils import boundary from kivymd import uix_path @@ -953,11 +952,11 @@ from kivymd.icon_definitions import md_icons from kivymd.theming import ThemableBehavior, ThemeManager from kivymd.uix.behaviors import ( DeclarativeBehavior, - FakeRectangularElevationBehavior, RectangularRippleBehavior, SpecificBackgroundColorBehavior, ) from kivymd.uix.boxlayout import MDBoxLayout +from kivymd.uix.card import MDCard from kivymd.uix.carousel import MDCarousel from kivymd.uix.label import MDLabel @@ -1024,7 +1023,7 @@ class MDTabsLabel(ToggleButtonBehavior, RectangularRippleBehavior, MDLabel): Clock.schedule_once(self.tab_bar._label_request_indicator_update, 0) -class MDTabsBase(Widget): +class MDTabsBase: """ This class allow you to create a tab. You must create a new class that inherits from MDTabsBase. @@ -1130,9 +1129,9 @@ class MDTabsBase(Widget): This property will affect the Tab's Title Label widget. """ - def __init__(self, **kwargs): + def __init__(self, *args, **kwargs): self.tab_label = MDTabsLabel(tab=self) - super().__init__(**kwargs) + super().__init__(*args, **kwargs) self.bind( icon=self._update_text, title=self._update_text, @@ -1273,9 +1272,7 @@ class MDTabsScrollView(ScrollView): _update(self.effect_y, scroll_y) -class MDTabsBar( - ThemableBehavior, FakeRectangularElevationBehavior, MDBoxLayout -): +class MDTabsBar(MDCard): """ This class is just a boxlayout that contains the scroll view for tabs. It is also responsible for resizing the tab shortcut when necessary. @@ -1551,13 +1548,43 @@ class MDTabs( and defaults to `None`. """ + shadow_softness = NumericProperty(12) + """ + See :attr:`kivymd.uix.behaviors.CommonElevationBehavior.shadow_softness` + attribute. + + .. versionadded:: 1.1.0 + + :attr:`shadow_softness` is an :class:`~kivy.properties.NumericProperty` + and defaults to `12`. + """ + + shadow_color = ColorProperty([0, 0, 0, 0.6]) + """ + See :attr:`kivymd.uix.behaviors.CommonElevationBehavior.shadow_color` + attribute. + + .. versionadded:: 1.1.0 + + :attr:`shadow_color` is an :class:`~kivy.properties.ColorProperty` + and defaults to `[0, 0, 0, 0.6]`. + """ + + shadow_offset = ListProperty((0, 0)) + """ + See :attr:`kivymd.uix.behaviors.CommonElevationBehavior.shadow_offset` + attribute. + + .. versionadded:: 1.1.0 + + :attr:`shadow_offset` is an :class:`~kivy.properties.ListProperty` + and defaults to `[0, 0]`. + """ + elevation = NumericProperty(0) """ - Tab value elevation. - - .. seealso:: - - `Behaviors/Elevation `_ + See :attr:`kivymd.uix.behaviors.CommonElevationBehavior.elevation` + attribute. :attr:`elevation` is an :class:`~kivy.properties.NumericProperty` and defaults to `0`. diff --git a/sbapp/kivymd/uix/taptargetview.py b/sbapp/kivymd/uix/taptargetview.py index 1e57ff0..735de0d 100644 --- a/sbapp/kivymd/uix/taptargetview.py +++ b/sbapp/kivymd/uix/taptargetview.py @@ -487,6 +487,8 @@ class MDTapTargetView(ThemableBehavior, EventDispatcher): _outer_radius = NumericProperty(0) _target_radius = NumericProperty(0) + __elevation = 0 + def __init__(self, **kwargs): self.ripple_max_dist = dp(90) self.on_outer_radius(self, self.outer_radius) @@ -514,6 +516,89 @@ class MDTapTargetView(ThemableBehavior, EventDispatcher): if not self.outer_circle_color: self.outer_circle_color = self.theme_cls.primary_color[:-1] + def start(self, *args): + """Starts widget opening animation.""" + + self._initialize() + self._animate_outer() + self.state = "open" + self.core_title_text.opacity = 1 + self.core_description_text.opacity = 1 + self.dispatch("on_open") + + elevation = getattr(self.widget, "elevation", None) + if elevation: + self.__elevation = elevation + self.widget.elevation = 0 + + def stop(self, *args): + """Starts widget close animation.""" + + # It needs a better implementation. + if self.anim_ripple is not None: + self.anim_ripple.unbind(on_complete=self._repeat_ripple) + self.core_title_text.opacity = 0 + self.core_description_text.opacity = 0 + anim = Animation( + d=0.15, + t="in_cubic", + **dict( + zip( + ["_outer_radius", "_target_radius", "target_ripple_radius"], + [0, 0, 0], + ) + ), + ) + anim.bind(on_complete=self._after_stop) + anim.start(self.widget) + + def on_open(self, *args): + """Called at the time of the start of the widget opening animation.""" + + def on_close(self, *args): + """Called at the time of the start of the widget closed animation.""" + + def on_draw_shadow(self, instance, value): + Logger.warning( + "The shadow adding method will be implemented in future versions" + ) + + def on_description_text(self, instance, value): + self.core_description_text.text = value + + def on_description_text_size(self, instance, value): + self.core_description_text.font_size = value + + def on_description_text_bold(self, instance, value): + self.core_description_text.bold = value + + def on_title_text(self, instance, value): + self.core_title_text.text = value + + def on_title_text_size(self, instance, value): + self.core_title_text.font_size = value + + def on_title_text_bold(self, instance, value): + self.core_title_text.bold = value + + def on_outer_radius(self, instance, value): + self._outer_radius = self.outer_radius * 2 + + def on_target_radius(self, instance, value): + self._target_radius = self.target_radius * 2 + + def on_target_touch(self): + if self.stop_on_target_touch: + self.stop() + + def on_outer_touch(self): + if self.stop_on_outer_touch: + self.stop() + + def on_outside_click(self): + if self.cancelable: + self.stop() + def _initialize(self): setattr(self.widget, "_outer_radius", 0) setattr(self.widget, "_target_radius", 0) @@ -527,7 +612,7 @@ class MDTapTargetView(ThemableBehavior, EventDispatcher): def _draw_canvas(self): _pos = self._ttv_pos() - self.widget.canvas.before.clear() + self.widget.canvas.before.remove_group("ttv_group") with self.widget.canvas.before: # Outer circle. @@ -588,34 +673,14 @@ class MDTapTargetView(ThemableBehavior, EventDispatcher): group="ttv_group", ) - def stop(self, *args): - """Starts widget close animation.""" - - # It needs a better implementation. - if self.anim_ripple is not None: - self.anim_ripple.unbind(on_complete=self._repeat_ripple) - self.core_title_text.opacity = 0 - self.core_description_text.opacity = 0 - anim = Animation( - d=0.15, - t="in_cubic", - **dict( - zip( - ["_outer_radius", "_target_radius", "target_ripple_radius"], - [0, 0, 0], - ) - ), - ) - anim.bind(on_complete=self._after_stop) - anim.start(self.widget) - def _after_stop(self, *args): self.widget.canvas.before.remove_group("ttv_group") args[0].stop_all(self.widget) - elev = getattr(self.widget, "elevation", None) - if elev: - self._fix_elev() + elevation = getattr(self.widget, "elevation", None) + if elevation: + self.widget.elevation = self.__elevation + self.dispatch("on_close") # Don't forget to unbind the function or it'll mess @@ -639,16 +704,6 @@ class MDTapTargetView(ThemableBehavior, EventDispatcher): ) Color(a=1) - def start(self, *args): - """Starts widget opening animation.""" - - self._initialize() - self._animate_outer() - self.state = "open" - self.core_title_text.opacity = 1 - self.core_description_text.opacity = 1 - self.dispatch("on_open") - def _animate_outer(self): anim = Animation( d=0.2, @@ -684,53 +739,6 @@ class MDTapTargetView(ThemableBehavior, EventDispatcher): setattr(self.widget, "target_ripple_alpha", 1) self._animate_ripple() - def on_open(self, *args): - """Called at the time of the start of the widget opening animation.""" - - def on_close(self, *args): - """Called at the time of the start of the widget closed animation.""" - - def on_draw_shadow(self, instance, value): - Logger.warning( - "The shadow adding method will be implemented in future versions" - ) - - def on_description_text(self, instance, value): - self.core_description_text.text = value - - def on_description_text_size(self, instance, value): - self.core_description_text.font_size = value - - def on_description_text_bold(self, instance, value): - self.core_description_text.bold = value - - def on_title_text(self, instance, value): - self.core_title_text.text = value - - def on_title_text_size(self, instance, value): - self.core_title_text.font_size = value - - def on_title_text_bold(self, instance, value): - self.core_title_text.bold = value - - def on_outer_radius(self, instance, value): - self._outer_radius = self.outer_radius * 2 - - def on_target_radius(self, instance, value): - self._target_radius = self.target_radius * 2 - - def on_target_touch(self): - if self.stop_on_target_touch: - self.stop() - - def on_outer_touch(self): - if self.stop_on_outer_touch: - self.stop() - - def on_outside_click(self): - if self.cancelable: - self.stop() - def _some_func(self, wid, touch): """ This function decides which one to dispatch based on the touch diff --git a/sbapp/kivymd/uix/templates/rotatewidget/rotatewidget.kv b/sbapp/kivymd/uix/templates/rotatewidget/rotatewidget.kv deleted file mode 100644 index 77b57df..0000000 --- a/sbapp/kivymd/uix/templates/rotatewidget/rotatewidget.kv +++ /dev/null @@ -1,9 +0,0 @@ - - canvas.before: - PushMatrix - Rotate: - angle: self.rotate_value_angle - axis: tuple(self.rotate_value_axis) - origin: self.center - canvas.after: - PopMatrix diff --git a/sbapp/kivymd/uix/templates/rotatewidget/rotatewidget.py b/sbapp/kivymd/uix/templates/rotatewidget/rotatewidget.py index d3491df..9416a5f 100644 --- a/sbapp/kivymd/uix/templates/rotatewidget/rotatewidget.py +++ b/sbapp/kivymd/uix/templates/rotatewidget/rotatewidget.py @@ -2,127 +2,31 @@ Templates/RotateWidget ====================== -.. versionadded:: 1.0.0 +.. deprecated:: 1.0.0 -Base class for controlling the rotate of the widget. - -.. note:: See `kivy.graphics.Rotate - `_ - for more information. - -Kivy ----- - -.. code-block:: python - - from kivy.animation import Animation - from kivy.lang import Builder - from kivy.app import App - from kivy.properties import NumericProperty - from kivy.uix.button import Button - - KV = ''' - Screen: - - RotateButton: - size_hint: .5, .5 - pos_hint: {"center_x": .5, "center_y": .5} - on_release: app.change_rotate(self) - - canvas.before: - PushMatrix - Rotate: - angle: self.rotate_value_angle - axis: 0, 0, 1 - origin: self.center - canvas.after: - PopMatrix - ''' - - - class RotateButton(Button): - rotate_value_angle = NumericProperty(0) - - - class Test(App): - def build(self): - return Builder.load_string(KV) - - def change_rotate(self, instance_button: Button) -> None: - Animation(rotate_value_angle=45, d=0.3).start(instance_button) - - - Test().run() - -KivyMD ------- - -.. code-block:: python - - from kivy.animation import Animation - from kivy.lang import Builder - - from kivymd.app import MDApp - from kivymd.uix.button import MDRaisedButton - from kivymd.uix.templates import RotateWidget - - KV = ''' - MDScreen: - - RotateButton: - size_hint: .5, .5 - pos_hint: {"center_x": .5, "center_y": .5} - on_release: app.change_rotate(self) - elevation:0 - ''' - - - class RotateButton(MDRaisedButton, RotateWidget): - pass - - - class Test(MDApp): - def build(self): - return Builder.load_string(KV) - - def change_rotate(self, instance_button: MDRaisedButton) -> None: - Animation(rotate_value_angle=45, d=0.3).start(instance_button) - - - Test().run() +.. note:: `RotateWidget` class has been deprecated. Please use + `RotateBahavior `_ + class instead. """ __all__ = ("RotateWidget",) -import os +from kivy import Logger -from kivy.lang import Builder -from kivy.properties import ListProperty, NumericProperty - -from kivymd import uix_path - -with open( - os.path.join(uix_path, "templates", "rotatewidget", "rotatewidget.kv"), - encoding="utf-8", -) as kv_file: - Builder.load_string(kv_file.read()) +from kivymd.uix.behaviors import RotateBehavior -class RotateWidget: - """Base class for controlling the rotate of the widget.""" - - rotate_value_angle = NumericProperty(0) +class RotateWidget(RotateBehavior): """ - Property for getting/setting the angle of the rotation. - - :attr:`rotate_value_angle` is an :class:`~kivy.properties.NumericProperty` - and defaults to `0`. + .. deprecated:: 1.1.0 + Use :class:`~kivymd.uix.behaviors.rotate_behavior.RotateBehavior` + class instead. """ - rotate_value_axis = ListProperty((0, 0, 1)) - """ - Property for getting/setting the axis of the rotation. - - :attr:`rotate_value_axis` is an :class:`~kivy.properties.NumericProperty` - and defaults to `(0, 0, 1)`. - """ + def __init__(self, **kwargs): + super().__init__(**kwargs) + Logger.warning( + "KivyMD: " + "The `RotateWidget` class has been deprecated. " + "Use the `RotateBehavior` class instead." + ) diff --git a/sbapp/kivymd/uix/templates/scalewidget/scalewidget.kv b/sbapp/kivymd/uix/templates/scalewidget/scalewidget.kv deleted file mode 100644 index 5fe002f..0000000 --- a/sbapp/kivymd/uix/templates/scalewidget/scalewidget.kv +++ /dev/null @@ -1,10 +0,0 @@ - - canvas.before: - PushMatrix - Scale: - x: self.scale_value_x - y: self.scale_value_y - z: self.scale_value_x - origin: self.center - canvas.after: - PopMatrix diff --git a/sbapp/kivymd/uix/templates/scalewidget/scalewidget.py b/sbapp/kivymd/uix/templates/scalewidget/scalewidget.py index 9f9204f..3fe6985 100644 --- a/sbapp/kivymd/uix/templates/scalewidget/scalewidget.py +++ b/sbapp/kivymd/uix/templates/scalewidget/scalewidget.py @@ -2,149 +2,33 @@ Templates/ScaleWidget ===================== -.. versionadded:: 1.0.0 +.. deprecated:: 1.1.0 Base class for controlling the scale of the widget. -.. note:: See `kivy.graphics.Scale - `_ - for more information. - -Kivy ----- - -.. code-block:: python - - from kivy.animation import Animation - from kivy.lang import Builder - from kivy.properties import NumericProperty - from kivy.uix.button import Button - from kivy.app import App - - - KV = ''' - Screen: - - ScaleButton: - size_hint: .5, .5 - pos_hint: {"center_x": .5, "center_y": .5} - on_release: app.change_scale(self) - - canvas.before: - PushMatrix - Scale: - x: self.scale_value_x - y: self.scale_value_y - z: self.scale_value_x - origin: self.center - canvas.after: - PopMatrix - ''' - - - class ScaleButton(Button): - scale_value_x = NumericProperty(1) - scale_value_y = NumericProperty(1) - scale_value_z = NumericProperty(1) - - - class Test(App): - def build(self): - return Builder.load_string(KV) - - def change_scale(self, instance_button: Button) -> None: - Animation( - scale_value_x=0.5, - scale_value_y=0.5, - scale_value_z=0.5, - d=0.3, - ).start(instance_button) - - - Test().run() - -KivyMD ------- - -.. code-block:: python - - from kivy.animation import Animation - from kivy.lang import Builder - - from kivymd.app import MDApp - from kivymd.uix.button import MDRaisedButton - from kivymd.uix.templates import ScaleWidget - - KV = ''' - MDScreen: - - ScaleButton: - size_hint: .5, .5 - pos_hint: {"center_x": .5, "center_y": .5} - on_release: app.change_scale(self) - elevation:0 - ''' - - - class ScaleButton(MDRaisedButton, ScaleWidget): - pass - - - class Test(MDApp): - def build(self): - return Builder.load_string(KV) - - def change_scale(self, instance_button: MDRaisedButton) -> None: - Animation( - scale_value_x=0.5, - scale_value_y=0.5, - scale_value_z=0.5, - d=0.3, - ).start(instance_button) - - - Test().run() +.. note:: `ScaleWidget` class has been deprecated. Please use + `ScaleBehavior `_ + class instead. """ __all__ = ("ScaleWidget",) -import os +from kivy import Logger -from kivy.lang import Builder -from kivy.properties import NumericProperty - -from kivymd import uix_path - -with open( - os.path.join(uix_path, "templates", "scalewidget", "scalewidget.kv"), - encoding="utf-8", -) as kv_file: - Builder.load_string(kv_file.read()) +from kivymd.uix.behaviors import ScaleBehavior -class ScaleWidget: - """Base class for controlling the scale of the widget.""" - - scale_value_x = NumericProperty(1) +class ScaleWidget(ScaleBehavior): """ - X-axis value. - - :attr:`scale_value_x` is an :class:`~kivy.properties.NumericProperty` - and defaults to `1`. + .. deprecated:: 1.1.0 + Use :class:`~kivymd.uix.behaviors.scale_behavior.ScaleBehavior` + class instead. """ - scale_value_y = NumericProperty(1) - """ - Y-axis value. - - :attr:`scale_value_y` is an :class:`~kivy.properties.NumericProperty` - and defaults to `1`. - """ - - scale_value_z = NumericProperty(1) - """ - Z-axis value. - - :attr:`scale_value_z` is an :class:`~kivy.properties.NumericProperty` - and defaults to `1`. - """ + def __init__(self, **kwargs): + super().__init__(**kwargs) + Logger.warning( + "KivyMD: " + "The `ScaleWidget` class has been deprecated. " + "Use the `ScaleBehavior` class instead." + ) diff --git a/sbapp/kivymd/uix/templates/stencilwidget/stencilwidget.kv b/sbapp/kivymd/uix/templates/stencilwidget/stencilwidget.kv deleted file mode 100644 index 04807dc..0000000 --- a/sbapp/kivymd/uix/templates/stencilwidget/stencilwidget.kv +++ /dev/null @@ -1,19 +0,0 @@ - - canvas.before: - StencilPush - RoundedRectangle: - pos: root.pos - size: root.size - # FIXME: Sometimes the radius has the value [], which get a - # `GraphicException: Invalid radius value, must be list of tuples/numerics` error - radius: root.radius if root.radius else [0, 0, 0, 0] - StencilUse - canvas.after: - StencilUnUse - RoundedRectangle: - pos: root.pos - size: root.size - # FIXME: Sometimes the radius has the value [], which get a - # `GraphicException: Invalid radius value, must be list of tuples/numerics` error - radius: root.radius if root.radius else [0, 0, 0, 0] - StencilPop diff --git a/sbapp/kivymd/uix/templates/stencilwidget/stencilwidget.py b/sbapp/kivymd/uix/templates/stencilwidget/stencilwidget.py index 3007857..a231588 100644 --- a/sbapp/kivymd/uix/templates/stencilwidget/stencilwidget.py +++ b/sbapp/kivymd/uix/templates/stencilwidget/stencilwidget.py @@ -2,115 +2,33 @@ Templates/StencilWidget ======================= -.. versionadded:: 1.0.0 +.. deprecated:: 1.1.0 Base class for controlling the stencil instructions of the widget. -.. note:: See `Stencil instructions - `_ - for more information. - -Kivy ----- - -.. code-block:: python - - from kivy.lang import Builder - from kivy.app import App - - KV = ''' - Carousel: - - Button: - size_hint: .9, .8 - pos_hint: {"center_x": .5, "center_y": .5} - - canvas.before: - StencilPush - RoundedRectangle: - pos: root.pos - size: root.size - StencilUse - canvas.after: - StencilUnUse - RoundedRectangle: - pos: root.pos - size: root.size - StencilPop - ''' - - - class Test(App): - def build(self): - return Builder.load_string(KV) - - - Test().run() - -KivyMD ------- - -.. code-block:: python - - from kivy.lang import Builder - - from kivymd.app import MDApp - from kivymd.uix.templates import StencilWidget - from kivymd.uix.fitimage import FitImage - - KV = ''' - MDCarousel: - - StencilImage: - size_hint: .9, .8 - pos_hint: {"center_x": .5, "center_y": .5} - source: "image.png" - ''' - - - class StencilImage(FitImage, StencilWidget): - pass - - - class Test(MDApp): - def build(self): - return Builder.load_string(KV) - - - Test().run() +.. note:: `StencilWidget` class has been deprecated. Please use + `StencilBehavior `_ + class instead. """ __all__ = ("StencilWidget",) -import os +from kivy import Logger -from kivy.lang import Builder -from kivy.properties import VariableListProperty - -from kivymd import uix_path - -with open( - os.path.join(uix_path, "templates", "stencilwidget", "stencilwidget.kv"), - encoding="utf-8", -) as kv_file: - Builder.load_string(kv_file.read()) +from kivymd.uix.behaviors import StencilBehavior -class StencilWidget: - """Base class for controlling the stencil instructions of the widget""" - - radius = VariableListProperty([0], length=4) +class StencilWidget(StencilBehavior): """ - Canvas radius. - - .. versionadded:: 1.0.0 - - .. code-block:: python - - # Top left corner slice. - MDWidget: - radius: [25, 0, 0, 0] - - :attr:`radius` is an :class:`~kivy.properties.VariableListProperty` - and defaults to `[0, 0, 0, 0]`. + .. deprecated:: 1.1.0 + Use :class:`~kivymd.uix.behaviors.scale_behavior.StencilBehavior` + class instead. """ + + def __init__(self, **kwargs): + super().__init__(**kwargs) + Logger.warning( + "KivyMD: " + "The `StencilWidget` class has been deprecated. " + "Use the `StencilBehavior` class instead." + ) diff --git a/sbapp/kivymd/uix/textfield/textfield.kv b/sbapp/kivymd/uix/textfield/textfield.kv index d1b5fc3..6f11abb 100644 --- a/sbapp/kivymd/uix/textfield/textfield.kv +++ b/sbapp/kivymd/uix/textfield/textfield.kv @@ -1,4 +1,7 @@ + input_filter: self.field_filter + do_backspace: self.do_backspace + canvas.before: Clear diff --git a/sbapp/kivymd/uix/textfield/textfield.py b/sbapp/kivymd/uix/textfield/textfield.py index f68bb90..08ea932 100755 --- a/sbapp/kivymd/uix/textfield/textfield.py +++ b/sbapp/kivymd/uix/textfield/textfield.py @@ -14,7 +14,6 @@ Components/TextField `KivyMD` provides the following field classes for use: - MDTextField_ -- MDTextFieldRound_ - MDTextFieldRect_ .. Note:: :class:`~MDTextField` inherited from @@ -79,15 +78,15 @@ parameter to `True`: from kivymd.app import MDApp KV = ''' - BoxLayout: - padding: "10dp" + MDScreen: MDTextField: id: text_field_error hint_text: "Helper text on error (press 'Enter')" helper_text: "There will always be a mistake" helper_text_mode: "on_error" - pos_hint: {"center_y": .5} + pos_hint: {"center_x": .5, "center_y": .5} + size_hint_x: .5 ''' @@ -97,6 +96,8 @@ parameter to `True`: self.screen = Builder.load_string(KV) def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" self.screen.ids.text_field_error.bind( on_text_validate=self.set_error_message, on_focus=self.set_error_message, @@ -119,6 +120,7 @@ Helper text mode `'on_error'` (with required) MDTextField: hint_text: "required = True" + text: "required = True" required: True helper_text_mode: "on_error" helper_text: "Enter text" @@ -186,7 +188,7 @@ Round mode max_text_length: 15 helper_text: "Massage" -.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-round-mode.png +.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-round-mode.gif :align: center .. MDTextFieldRect: @@ -203,6 +205,7 @@ MDTextFieldRect MDTextFieldRect: size_hint: 1, None height: "30dp" + background_color: app.theme_cls.bg_normal .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-rect.gif :align: center @@ -278,18 +281,17 @@ __all__ = ("MDTextField", "MDTextFieldRect") import os import re +from datetime import date from typing import Union from kivy.animation import Animation from kivy.clock import Clock from kivy.lang import Builder -from kivy.logger import Logger from kivy.metrics import dp, sp from kivy.properties import ( AliasProperty, BooleanProperty, ColorProperty, - DictProperty, ListProperty, NumericProperty, ObjectProperty, @@ -311,6 +313,220 @@ with open( Builder.load_string(kv_file.read()) +# TODO: Add a class to work with the phone number mask. + + +class AutoFormatTelephoneNumber: + """ + Implements automatic formatting of the text entered in the text field + according to the mask, for example '+38 (###) ### ## ##'. + """ + + def __init__(self): + self._backspace = False + + def isnumeric(self, value): + try: + int(value) + return True + except ValueError: + return False + + def do_backspace(self, *args): + if self.validator and self.validator == "phone": + self._backspace = True + text = self.text + text = text[:-1] + self.text = text + self._backspace = False + + def field_filter(self, value, boolean): + if self.validator and self.validator == "phone": + if len(self.text) == 14: + return + if self.isnumeric(value): + return value + return value + + def format(self, value): + if value != "" and not value.isspace() and not self._backspace: + if len(value) <= 1 and self.focus: + self.text = value + self._check_cursor() + elif len(value) == 4: + start = self.text[:-1] + end = self.text[-1] + self.text = "%s) %s" % (start, end) + self._check_cursor() + elif len(value) == 8: + self.text += "-" + self._check_cursor() + elif len(value) in [12, 16]: + start = self.text[:-1] + end = self.text[-1] + self.text = "%s-%s" % (start, end) + self._check_cursor() + + def _check_cursor(self): + def set_pos_cursor(pos_corsor, interval=0.5): + self.cursor = (pos_corsor, 0) + + if self.focus: + Clock.schedule_once(lambda x: set_pos_cursor(len(self.text)), 0.1) + + +class Validator: + """Container class for various validation methods.""" + + datetime_date = ObjectProperty() + """ + The last valid date as a object. + + :attr:`datetime_date` is an :class:`~kivy.properties.ObjectProperty` + and defaults to `None`. + """ + + date_interval = ListProperty([None, None]) + """ + The date interval that is valid for input. + Can be entered as objects or a string format. + Both values or just one value can be entered. + + In string format, must follow the current date_format. + Example: Given date_format -> "mm/dd/yyyy" + Input examples -> "12/31/1900", "12/31/2100" or "12/31/1900", None. + + :attr:`date_interval` is an :class:`~kivy.properties.ListProperty` + and defaults to `[None, None]`. + """ + + date_format = OptionProperty( + None, + options=[ + "dd/mm/yyyy", + "mm/dd/yyyy", + "yyyy/mm/dd", + ], + ) + + """ + Format of date strings that will be entered. + Available options are: `'dd/mm/yyyy'`, `'mm/dd/yyyy'`, `'yyyy/mm/dd'`. + + :attr:`date_format` is an :class:`~kivy.properties.OptionProperty` + and defaults to `None`. + """ + + def is_email_valid(self, text: str) -> bool: + if not re.match(r"[^@]+@[^@]+\.[^@]+", text): + return True + return False + + def is_time_valid(self, text: str) -> bool: + if re.match(r"^(2[0-3]|[01]?[0-9]):([0-5]?[0-9])$", text) or re.match( + r"^(2[0-3]|[01]?[0-9]):([0-5]?[0-9]):([0-5]?[0-9])$", text + ): + return False + + return True + + def is_date_valid(self, text: str) -> bool: + if not self.date_format: + raise Exception("TextInput date_format was not defined.") + + # Regex strings. + dd = "[0][1-9]|[1-2][0-9]|[3][0-1]" + mm = "[0][1-9]|[1][0-2]" + yyyy = "[0-9][0-9][0-9][0-9]" + fmt = self.date_format.split("/") + largs = locals() + # Access the local variables dict in the correct format based on + # date_format split. Example: "mm/dd/yyyy" -> ["mm", "dd", "yyyy"] + # largs[fmt[0]] would be largs["mm"] so the month regex string. + if re.match( + f"^({largs[fmt[0]]})/({largs[fmt[1]]})/({largs[fmt[2]]})$", text + ): + input_split = text.split("/") + largs[fmt[0]] = input_split[0] + largs[fmt[1]] = input_split[1] + largs[fmt[2]] = input_split[2] + # Organize input into correct slots and try to convert + # to datetime object. This way February exceptions are + # tested. Also tests with the date_interval are simpler + # using datetime objects. + try: + datetime = date( + int(largs["yyyy"]), int(largs["mm"]), int(largs["dd"]) + ) + except ValueError: + return True + + if self.date_interval: + if ( + self.date_interval[0] + and not self.date_interval[0] <= datetime + or self.date_interval[1] + and not datetime <= self.date_interval[1] + ): + return True + + self.datetime_date = datetime + return False + return True + + def on_date_interval(self, *args) -> None: + """Default event handler for date_interval input.""" + + def on_date_interval(): + if not self.date_format: + raise Exception("TextInput date_format was not defined.") + + fmt = self.date_format.split("/") + largs = {} + # Convert string inputs into datetime.date objects and store + # them back into self.date_interval. + try: + if self.date_interval[0] and not isinstance( + self.date_interval[0], date + ): + split = self.date_interval[0].split("/") + largs[fmt[0]] = split[0] + largs[fmt[1]] = split[1] + largs[fmt[2]] = split[2] + self.date_interval[0] = date( + int(largs["yyyy"]), int(largs["mm"]), int(largs["dd"]) + ) + if self.date_interval[1] and not isinstance( + self.date_interval[1], date + ): + split = self.date_interval[1].split("/") + largs[fmt[0]] = split[0] + largs[fmt[1]] = split[1] + largs[fmt[2]] = split[2] + self.date_interval[1] = date( + int(largs["yyyy"]), int(largs["mm"]), int(largs["dd"]) + ) + + except Exception: + raise Exception( + r"TextInput date_interval was defined incorrectly, it must " + r"be composed of objects or strings" + r" following current date_format." + ) + + # Test if the interval is valid. + if isinstance(self.date_interval[0], date) and isinstance( + self.date_interval[1], date + ): + if self.date_interval[0] >= self.date_interval[1]: + raise Exception( + "TextInput date_interval last date must be greater" + " than the first date or set to None." + ) + + Clock.schedule_once(lambda x: on_date_interval()) + + class MDTextFieldRect(ThemableBehavior, TextInput): line_anim = BooleanProperty(True) """ @@ -383,7 +599,13 @@ class TextfieldLabel(ThemableBehavior, Label): self.font_size = sp(self.theme_cls.font_styles[self.font_style][1]) -class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): +class MDTextField( + DeclarativeBehavior, + ThemableBehavior, + TextInput, + Validator, + AutoFormatTelephoneNumber, +): helper_text = StringProperty() """ Text for ``helper_text`` mode. @@ -430,17 +652,185 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): and defaults to `'line'`. """ + phone_mask = StringProperty("") + + validator = OptionProperty(None, options=["date", "email", "time", "phone"]) + """ + The type of text field for entering Email, time, etc. + Automatically sets the type of the text field as "error" if the user input + does not match any of the set validation types. + Available options are: `'date'`, `'email'`, `'time'`. + + When using `'date'`, :attr:`date_format` must be defined. + + .. versionadded:: 1.1.0 + + .. code-block:: python + + MDTextField: + hint_text: "Email" + helper_text: "user@gmail.com" + validator: "email" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-validator.png + :align: center + + .. tabs:: + + .. tab:: Declarative KV style + + .. code-block:: python + + from kivy.lang import Builder + + from kivymd.app import MDApp + + KV = ''' + MDScreen: + + MDBoxLayout: + orientation: "vertical" + spacing: "20dp" + adaptive_height: True + size_hint_x: .8 + pos_hint: {"center_x": .5, "center_y": .5} + + MDTextField: + hint_text: "Date dd/mm/yyyy without limits" + helper_text: "Enter a valid dd/mm/yyyy date" + validator: "date" + date_format: "dd/mm/yyyy" + + MDTextField: + hint_text: "Date mm/dd/yyyy without limits" + helper_text: "Enter a valid mm/dd/yyyy date" + validator: "date" + date_format: "mm/dd/yyyy" + + MDTextField: + hint_text: "Date yyyy/mm/dd without limits" + helper_text: "Enter a valid yyyy/mm/dd date" + validator: "date" + date_format: "yyyy/mm/dd" + + MDTextField: + hint_text: "Date dd/mm/yyyy in [01/01/1900, 01/01/2100] interval" + helper_text: "Enter a valid dd/mm/yyyy date" + validator: "date" + date_format: "dd/mm/yyyy" + date_interval: "01/01/1900", "01/01/2100" + + MDTextField: + hint_text: "Date dd/mm/yyyy in [01/01/1900, None] interval" + helper_text: "Enter a valid dd/mm/yyyy date" + validator: "date" + date_format: "dd/mm/yyyy" + date_interval: "01/01/1900", None + + MDTextField: + hint_text: "Date dd/mm/yyyy in [None, 01/01/2100] interval" + helper_text: "Enter a valid dd/mm/yyyy date" + validator: "date" + date_format: "dd/mm/yyyy" + date_interval: None, "01/01/2100" + ''' + + + class Test(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return Builder.load_string(KV) + + + Test().run() + + .. tab:: Declarative python style + + .. code-block:: python + + from kivymd.app import MDApp + from kivymd.uix.boxlayout import MDBoxLayout + from kivymd.uix.screen import MDScreen + from kivymd.uix.textfield import MDTextField + + + class Test(MDApp): + def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" + return ( + MDScreen( + MDBoxLayout( + MDTextField( + hint_text="Date dd/mm/yyyy without limits", + helper_text="Enter a valid dd/mm/yyyy date", + validator="date", + date_format="dd/mm/yyyy", + ), + MDTextField( + hint_text="Date mm/dd/yyyy without limits", + helper_text="Enter a valid mm/dd/yyyy date", + validator="date", + date_format="mm/dd/yyyy", + ), + MDTextField( + hint_text="Date yyyy/mm/dd without limits", + helper_text="Enter a valid yyyy/mm/dd date", + validator="date", + date_format="yyyy/mm/dd", + ), + MDTextField( + hint_text="Date dd/mm/yyyy in [01/01/1900, 01/01/2100] interval", + helper_text="Enter a valid dd/mm/yyyy date", + validator="date", + date_format="dd/mm/yyyy", + date_interval=["01/01/1900", "01/01/2100"], + ), + MDTextField( + hint_text="Date dd/mm/yyyy in [01/01/1900, None] interval", + helper_text="Enter a valid dd/mm/yyyy date", + validator="date", + date_format="dd/mm/yyyy", + date_interval=["01/01/1900", None], + ), + MDTextField( + hint_text="Date dd/mm/yyyy in [None, 01/01/2100] interval", + helper_text="Enter a valid dd/mm/yyyy date", + validator="date", + date_format="dd/mm/yyyy", + date_interval=[None, "01/01/2100"], + ), + orientation="vertical", + spacing="20dp", + adaptive_height=True, + size_hint_x=0.8, + pos_hint={"center_x": 0.5, "center_y": 0.5}, + ) + ) + ) + + + Test().run() + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-validator-date.png + :align: center + + :attr:`validator` is an :class:`~kivy.properties.OptionProperty` + and defaults to `None`. + """ + line_color_normal = ColorProperty([0, 0, 0, 0]) """ - Line color normal (static underline line) in ``rgba`` format. + Line color normal (static underline line) in (r, g, b, a) or string format. .. code-block:: kv MDTextField: hint_text: "line_color_normal" - line_color_normal: 1, 0, 1, 1 + line_color_normal: "red" - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-line-color-normal.gif + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-line-color-normal.png :align: center :attr:`line_color_normal` is an :class:`~kivy.properties.ColorProperty` @@ -449,13 +839,13 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): line_color_focus = ColorProperty([0, 0, 0, 0]) """ - Line color focus (active underline line) in ``rgba`` format. + Line color focus (active underline line) in (r, g, b, a) or string format. .. code-block:: kv MDTextField: hint_text: "line_color_focus" - line_color_focus: 0, 1, 0, 1 + line_color_focus: "red" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-line-color-focus.gif :align: center @@ -474,7 +864,7 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): error_color = ColorProperty([0, 0, 0, 0]) """ - Error color in ``rgba`` format for ``required = True``. + Error color in (r, g, b, a) or string format for ``required = True``. :attr:`error_color` is an :class:`~kivy.properties.ColorProperty` and defaults to `[0, 0, 0, 0]`. @@ -482,7 +872,18 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): fill_color_normal = ColorProperty([0, 0, 0, 0]) """ - Fill background color in 'fill' mode when text field is out of focus. + Fill background color in (r, g, b, a) or string format in 'fill' mode when] + text field is out of focus. + + .. code=block:: kv + + MDTextField: + hint_text: "Fill mode" + mode: "fill" + fill_color_normal: "brown" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-fill-color-normal.png + :align: center :attr:`fill_color_normal` is an :class:`~kivy.properties.ColorProperty` and defaults to `[0, 0, 0, 0]`. @@ -490,7 +891,18 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): fill_color_focus = ColorProperty([0, 0, 0, 0]) """ - Fill background color in 'fill' mode when the text field has focus. + Fill background color in (r, g, b, a) or string format in 'fill' mode when + the text field has focus. + + .. code=block:: kv + + MDTextField: + hint_text: "Fill mode" + mode: "fill" + fill_color_focus: "brown" + + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-fill-color-focus.gif + :align: center :attr:`fill_color_focus` is an :class:`~kivy.properties.ColorProperty` and defaults to `[0, 0, 0, 0]`. @@ -514,7 +926,8 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): hint_text_color_normal = ColorProperty([0, 0, 0, 0]) """ - Hint text color when text field is out of focus. + Hint text color in (r, g, b, a) or string format when text field is out + of focus. .. versionadded:: 1.0.0 @@ -522,9 +935,9 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: hint_text: "hint_text_color_normal" - hint_text_color_normal: 0, 1, 0, 1 + hint_text_color_normal: "red" - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-hint-text-color-normal.gif + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-hint-text-color-normal.png :align: center :attr:`hint_text_color_normal` is an :class:`~kivy.properties.ColorProperty` @@ -533,7 +946,8 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): hint_text_color_focus = ColorProperty([0, 0, 0, 0]) """ - Hint text color when the text field has focus. + Hint text color in (r, g, b, a) or string format when the text field has + focus. .. versionadded:: 1.0.0 @@ -541,7 +955,7 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: hint_text: "hint_text_color_focus" - hint_text_color_focus: 0, 1, 0, 1 + hint_text_color_focus: "red" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-hint-text-color-focus.gif :align: center @@ -552,7 +966,8 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): helper_text_color_normal = ColorProperty([0, 0, 0, 0]) """ - Helper text color when text field is out of focus. + Helper text color in (r, g, b, a) or string format when text field is out + of focus. .. versionadded:: 1.0.0 @@ -561,7 +976,7 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: helper_text: "helper_text_color_normal" helper_text_mode: "persistent" - helper_text_color_normal: 0, 1, 0, 1 + helper_text_color_normal: "red" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-helper-text-color-normal.png :align: center @@ -572,7 +987,8 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): helper_text_color_focus = ColorProperty([0, 0, 0, 0]) """ - Helper text color when the text field has focus. + Helper text color in (r, g, b, a) or string format when the text field has + focus. .. versionadded:: 1.0.0 @@ -581,7 +997,7 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: helper_text: "helper_text_color_focus" helper_text_mode: "persistent" - helper_text_color_focus: 0, 1, 0, 1 + helper_text_color_focus: "red" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-helper-text-color-focus.gif :align: center @@ -592,7 +1008,8 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): icon_right_color_normal = ColorProperty([0, 0, 0, 0]) """ - Color of right icon when text field is out of focus. + Color in (r, g, b, a) or string format of right icon when text field is out + of focus. .. versionadded:: 1.0.0 @@ -601,9 +1018,9 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: icon_right: "language-python" hint_text: "icon_right_color_normal" - icon_right_color_normal: 0, 1, 0, 1 + icon_right_color_normal: "red" - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-icon-right-color-normal.gif + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-icon-right-color-normal.png :align: center :attr:`icon_right_color_normal` is an :class:`~kivy.properties.ColorProperty` @@ -612,7 +1029,8 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): icon_right_color_focus = ColorProperty([0, 0, 0, 0]) """ - Color of right icon when the text field has focus. + Color in (r, g, b, a) or string format of right icon when the text field + has focus. .. versionadded:: 1.0.0 @@ -621,7 +1039,7 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: icon_right: "language-python" hint_text: "icon_right_color_focus" - icon_right_color_focus: 0, 1, 0, 1 + icon_right_color_focus: "red" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-icon-right-color-focus.gif :align: center @@ -632,47 +1050,30 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): icon_left_color_normal = ColorProperty([0, 0, 0, 0]) """ - Color of right icon when text field is out of focus. + Color in (r, g, b, a) or string format of right icon when text field is out + of focus. .. versionadded:: 1.0.0 - .. code-block:: kv - - MDTextField: - icon_right: "language-python" - hint_text: "icon_right_color_normal" - icon_left_color_normal: 0, 1, 0, 1 - - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-icon-right-color-normal.gif - :align: center - :attr:`icon_left_color_normal` is an :class:`~kivy.properties.ColorProperty` and defaults to `[0, 0, 0, 0]`. """ icon_left_color_focus = ColorProperty([0, 0, 0, 0]) """ - Color of right icon when the text field has focus. + Color in (r, g, b, a) or string format of right icon when the text field + has focus. .. versionadded:: 1.0.0 - .. code-block:: kv - - MDTextField: - icon_right: "language-python" - hint_text: "icon_right_color_focus" - icon_right_color_focus: 0, 1, 0, 1 - - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-icon-right-color-focus.gif - :align: center - :attr:`icon_left_color_focus` is an :class:`~kivy.properties.ColorProperty` and defaults to `[0, 0, 0, 0]`. """ max_length_text_color = ColorProperty([0, 0, 0, 0]) """ - Text color of the maximum length of characters to be input. + Text color in (r, g, b, a) or string format of the maximum length of + characters to be input. .. versionadded:: 1.0.0 @@ -680,10 +1081,10 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: hint_text: "max_length_text_color" - max_length_text_color: 0, 1, 0, 1 + max_length_text_color: "red" max_text_length: 5 - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-max-length-text-color.gif + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-max-length-text-color.png :align: center :attr:`max_length_text_color` is an :class:`~kivy.properties.ColorProperty` @@ -718,7 +1119,7 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): text_color_normal = ColorProperty([0, 0, 0, 0]) """ - Text color in ``rgba`` format when text field is out of focus. + Text color in (r, g, b, a) or string format when text field is out of focus. .. versionadded:: 1.0.0 @@ -726,9 +1127,9 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: hint_text: "text_color_normal" - text_color_normal: 0, 1, 0, 1 + text_color_normal: "red" - .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-text-color-normal.gif + .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-text-color-normal.png :align: center :attr:`text_color_normal` is an :class:`~kivy.properties.ColorProperty` @@ -737,7 +1138,7 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): text_color_focus = ColorProperty([0, 0, 0, 0]) """ - Text color in ``rgba`` format when text field has focus. + Text color in (r, g, b, a) or string format when text field has focus. .. versionadded:: 1.0.0 @@ -745,7 +1146,7 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): MDTextField: hint_text: "text_color_focus" - text_color_focus: 0, 1, 0, 1 + text_color_focus: "red" .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/text-field-text-color-focus.gif :align: center @@ -879,8 +1280,8 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): text=self.set_text, ) self.theme_cls.bind( - primary_color=lambda x, y: self.set_default_colors(0, True), - theme_style=lambda x, y: self.set_default_colors(0, True), + primary_color=self.set_default_colors, + theme_style=self.set_default_colors, ) Clock.schedule_once(self.check_text) @@ -930,9 +1331,17 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): ) if self.error_color == [0, 0, 0, 0] or updated: - self.error_color = self.theme_cls.error_color + self.error_color = ( + self.theme_cls.error_color + if self.error_color == [0, 0, 0, 0] + else self.error_color + ) if self.max_length_text_color == [0, 0, 0, 0] or updated: - self.max_length_text_color = self.theme_cls.disabled_hint_text_color + self.max_length_text_color = ( + self.theme_cls.disabled_hint_text_color + if self.max_length_text_color == [0, 0, 0, 0] + else self.max_length_text_color + ) self._hint_text_color = self.hint_text_color_normal self._text_color_normal = self.text_color_normal @@ -1101,8 +1510,11 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): self.text = re.sub("\n", " ", text) if not self.multiline else text self.set_max_text_length() + if self.validator and self.validator == "phone": + pass + # self.format(self.text) - if self.text and self.max_length_text_color and self._get_has_error(): + if (self.text and self.max_length_text_color) or self._get_has_error(): self.error = True if ( self.text @@ -1301,22 +1713,34 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): if value_height >= self.max_height and self.max_height: self.height = self.max_height - def on_text_color_normal(self, instance_text_field, color: list): + def on_text_color_normal( + self, instance_text_field, color: Union[list, str] + ): self._text_color_normal = color - def on_hint_text_color_normal(self, instance_text_field, color: list): + def on_hint_text_color_normal( + self, instance_text_field, color: Union[list, str] + ): self._hint_text_color = color - def on_helper_text_color_normal(self, instance_text_field, color: list): + def on_helper_text_color_normal( + self, instance_text_field, color: Union[list, str] + ): self._helper_text_color = color - def on_icon_right_color_normal(self, instance_text_field, color: list): + def on_icon_right_color_normal( + self, instance_text_field, color: Union[list, str] + ): self._icon_right_color = color - def on_line_color_normal(self, instance_text_field, color: list): + def on_line_color_normal( + self, instance_text_field, color: Union[list, str] + ): self._line_color_normal = color - def on_max_length_text_color(self, instance_text_field, color: list): + def on_max_length_text_color( + self, instance_text_field, color: Union[list, str] + ): self._max_length_text_color = color def _set_color(self, attr_name: str, color: str, updated: bool) -> None: @@ -1353,6 +1777,13 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): the :attr:`~MDTextField.required` parameter is set to `True`. """ + if self.validator and self.validator != "phone": + has_error = { + "date": self.is_date_valid, + "email": self.is_email_valid, + "time": self.is_time_valid, + }[self.validator](self.text) + return has_error if self.max_text_length and len(self.text) > self.max_text_length: has_error = True else: @@ -1367,9 +1798,12 @@ class MDTextField(DeclarativeBehavior, ThemableBehavior, TextInput): if __name__ == "__main__": + from kivy.core.window import Window from kivy.lang import Builder from kivy.uix.textinput import TextInput + Window.size = (800, 750) + from kivymd.app import MDApp KV = """ @@ -1385,41 +1819,53 @@ MDScreen: MDTextField: hint_text: "Label" - helper_text: "Error massage" + helper_text: "Error message" mode: "rectangle" max_text_length: 5 MDTextField: icon_left: "git" hint_text: "Label" - helper_text: "Error massage" + helper_text: "Error message" mode: "rectangle" MDTextField: icon_left: "git" hint_text: "Label" - helper_text: "Error massage" + helper_text: "Error message" mode: "fill" MDTextField: hint_text: "Label" - helper_text: "Error massage" + helper_text: "Error message" mode: "fill" MDTextField: hint_text: "Label" - helper_text: "Error massage" + helper_text: "Error message" MDTextField: icon_left: "git" hint_text: "Label" - helper_text: "Error massage" + helper_text: "Error message" MDTextField: hint_text: "Round mode" mode: "round" max_text_length: 15 - helper_text: "Massage" + helper_text: "Message" + + MDTextField: + hint_text: "Date dd/mm/yyyy in [01/01/1900, 01/01/2100] interval" + helper_text: "Enter a valid dd/mm/yyyy date" + validator: "date" + date_format: "dd/mm/yyyy" + date_interval: "01/01/1900", "01/01/2100" + + MDTextField: + hint_text: "Email" + helper_text: "user@gmail.com" + validator: "email" MDFlatButton: text: "SET TEXT" @@ -1429,6 +1875,8 @@ MDScreen: class Test(MDApp): def build(self): + self.theme_cls.theme_style = "Dark" + self.theme_cls.primary_palette = "Orange" return Builder.load_string(KV) def set_text(self): diff --git a/sbapp/kivymd/uix/toolbar/toolbar.py b/sbapp/kivymd/uix/toolbar/toolbar.py index a1ea57f..0a48548 100755 --- a/sbapp/kivymd/uix/toolbar/toolbar.py +++ b/sbapp/kivymd/uix/toolbar/toolbar.py @@ -121,8 +121,8 @@ Shadow elevation control .. code-block:: kv MDTopAppBar: - title: "Elevation 10" - elevation: 10 + title: "Elevation 4" + elevation: 4 .. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/toolbar-7.png :align: center @@ -327,7 +327,7 @@ Material design 3 style :align: center """ -__all__ = ("MDTopAppBar", "MDBottomAppBar") +__all__ = ("MDTopAppBar", "MDBottomAppBar", "ActionTopAppBarButton") import os from math import cos, radians, sin @@ -337,10 +337,8 @@ from kivy.animation import Animation from kivy.clock import Clock from kivy.core.window import Window from kivy.lang import Builder -from kivy.logger import Logger from kivy.metrics import dp from kivy.properties import ( - AliasProperty, BooleanProperty, ColorProperty, ListProperty, @@ -356,15 +354,15 @@ from kivymd import uix_path from kivymd.color_definitions import text_colors from kivymd.theming import ThemableBehavior from kivymd.uix.behaviors import ( + CommonElevationBehavior, DeclarativeBehavior, - FakeRectangularElevationBehavior, + ScaleBehavior, SpecificBackgroundColorBehavior, ) from kivymd.uix.button import MDFloatingActionButton, MDIconButton from kivymd.uix.controllers import WindowController from kivymd.uix.list import OneLineIconListItem from kivymd.uix.menu import MDDropdownMenu -from kivymd.uix.templates import ScaleWidget from kivymd.uix.tooltip import MDTooltip from kivymd.utils.set_bars_colors import set_bars_colors @@ -374,7 +372,7 @@ with open( Builder.load_string(kv_file.read()) -class ActionBottomAppBarButton(MDFloatingActionButton, ScaleWidget): +class ActionBottomAppBarButton(MDFloatingActionButton, ScaleBehavior): """ Implements a floating action button (FAB) for a toolbar with type 'bottom'. """ @@ -409,11 +407,11 @@ class OverFlowMenuItem(OneLineIconListItem): class NotchedBox( ThemableBehavior, - FakeRectangularElevationBehavior, + CommonElevationBehavior, SpecificBackgroundColorBehavior, BoxLayout, ): - elevation = NumericProperty(6) + elevation = NumericProperty(4) notch_radius = NumericProperty() notch_center_x = NumericProperty("100dp") @@ -961,8 +959,10 @@ class MDTopAppBar(DeclarativeBehavior, NotchedBox, WindowController): self.icon_color = self.theme_cls.primary_color self.bind(specific_text_color=self.update_action_bar_text_colors) - self.theme_cls.bind(material_style=self.update_bar_height) - self.theme_cls.bind(primary_palette=self.update_md_bg_color) + self.theme_cls.bind( + material_style=self.update_bar_height, + primary_palette=self.update_md_bg_color, + ) Clock.schedule_once( lambda x: self.on_left_action_items(0, self.left_action_items) @@ -1103,6 +1103,7 @@ class MDTopAppBar(DeclarativeBehavior, NotchedBox, WindowController): + self.theme_cls.standard_increment / 2 + self._shift ) + self.shadow_offset = [0, 30] self.on_mode(None, self.mode) def on_type_height(self, instance_toolbar, height_type_value: str) -> None: diff --git a/sbapp/kivymd/uix/tooltip/tooltip.py b/sbapp/kivymd/uix/tooltip/tooltip.py index 9b05c72..855119a 100644 --- a/sbapp/kivymd/uix/tooltip/tooltip.py +++ b/sbapp/kivymd/uix/tooltip/tooltip.py @@ -314,7 +314,7 @@ class MDTooltip(ThemableBehavior, HoverBehavior, TouchBehavior): Clock.schedule_once(self.animation_tooltip_dismiss) def on_show(self) -> None: - """Default dismiss event handler.""" + """Default display event handler.""" def on_dismiss(self) -> None: """ diff --git a/sbapp/kivymd/uix/transition/transition.py b/sbapp/kivymd/uix/transition/transition.py index 49bd657..169584d 100644 --- a/sbapp/kivymd/uix/transition/transition.py +++ b/sbapp/kivymd/uix/transition/transition.py @@ -35,7 +35,9 @@ __all__ = ( "MDTransitionBase", ) +from kivy import Logger from kivy.animation import Animation, AnimationTransition +from kivy.properties import DictProperty from kivy.uix.screenmanager import ( ScreenManagerException, SlideTransition, @@ -43,13 +45,41 @@ from kivy.uix.screenmanager import ( TransitionBase, ) +from kivymd.uix.hero import MDHeroFrom, MDHeroTo from kivymd.uix.screenmanager import MDScreenManager class MDTransitionBase(TransitionBase): + """ + TransitionBase is used to animate 2 screens within the + :class:`~kivymd.uix.screenmanager.MDScreenManager`. + + For more + information, see in the :class:`~kivy.uix.screenmanager.TransitionBase` + class documentation. + """ + _direction = "in" - hero_widget = None - hero_from_widget = None # kivymd.uix.hero.MDHeroFrom object + # Collection of child widgets of all 'MDHeroFrom' widgets that are + # on the screen, for example: + # + # MDScreen: + # + # MDHeroFrom: + # tag: "kivymd" + # + # FitImage: + # + # MDHeroFrom: + # tag: "kivy" + # + # FitImage: + # + # { + # 'kivy': , + # 'kivymd': , + # } + _hero_from_widget_children = DictProperty() def start(self, instance_screen_manager: MDScreenManager) -> None: super().start(instance_screen_manager) @@ -59,67 +89,160 @@ class MDTransitionBase(TransitionBase): ]() def animated_hero_in(self) -> None: - if self.manager._heroes_data and self.manager.current_hero: - self.hero_from_widget = self.manager.get_hero_from_widget() - self._check_widget_properties() - self.hero_widget = self.hero_from_widget.children[0] - self.hero_from_widget.remove_widget(self.hero_widget) + """Animates the flight of heroes from screen **A** to screen **B**.""" - self.hero_widget.pos = self.screen_out.to_widget( - *self.hero_from_widget.to_window(*self.hero_from_widget.pos) - ) - self.hero_widget.size = self.hero_from_widget.size - self.manager.get_root_window().add_widget(self.hero_widget) + if self.manager._heroes_data and self.manager.current_heroes: + for hero_from_widget in self.manager.get_hero_from_widget(): + for heroes_tag in self.manager.current_heroes: + if heroes_tag == hero_from_widget.tag: + self._check_widget_properties(hero_from_widget) - Animation( - size=self.screen_in.hero_to.size, - d=self.duration, - pos=self.screen_in.hero_to.pos, - ).start(self.hero_widget) - self.hero_from_widget.dispatch( - "on_transform_in", self.hero_widget, self.duration - ) + # Get child widget of the 'MDHeroFrom' container. + hero_widget = hero_from_widget.children[0] + self._hero_from_widget_children[ + hero_from_widget.tag + ] = hero_widget + + # Removing the child widget from the 'MDHeroFrom' + # container. + hero_from_widget.remove_widget(hero_widget) + + # We set the size, position of the child widget of the + # 'MDHeroFrom' container and add this widget to the + # root window. + hero_widget.pos = self.screen_out.to_widget( + *hero_from_widget.to_window(*hero_from_widget.pos) + ) + hero_widget.size = hero_from_widget.size + self.manager.get_root_window().add_widget(hero_widget) + + # Animating widgets added to the root window. + if self.screen_in.heroes_to: + for hero_to_widget in self.screen_in.heroes_to: + self._check_hero_to_widget_tag( + hero_to_widget, hero_from_widget + ) + if hero_to_widget.tag == heroes_tag: + Animation( + size=hero_to_widget.size, + d=self.duration, + pos=hero_to_widget.pos, + ).start(hero_widget) + hero_from_widget.dispatch( + "on_transform_in", + hero_widget, + self.duration, + ) def animated_hero_out(self) -> None: - if self.manager._heroes_data and self.manager.current_hero: - self.screen_out.hero_to.remove_widget(self.hero_widget) - self.manager.get_root_window().add_widget(self.hero_widget) + """Animates the flight of heroes from screen **B** to screen **A**.""" - self.hero_from_widget.dispatch( - "on_transform_out", self.hero_widget, self.duration - ) - Animation( - pos=self.screen_in.to_widget( - *self.hero_from_widget.to_window(*self.hero_from_widget.pos) - ), - size=self.hero_from_widget.size, - d=self.duration, - ).start(self.hero_widget) + if ( + self.manager._heroes_data + and self.manager.current_heroes + and self.screen_out.heroes_to + ): + + for heroes_tag in self.manager.current_heroes: + for hero_to_widget in self.screen_out.heroes_to: + if hero_to_widget.tag == heroes_tag: + hero_from_children = self._hero_from_widget_children[ + heroes_tag + ] + hero_to_widget.remove_widget(hero_from_children) + self.manager.get_root_window().add_widget( + hero_from_children + ) + + for ( + hero_from_widget + ) in self.manager.get_hero_from_widget(): + hero_from_widget.dispatch( + "on_transform_out", + self._hero_from_widget_children[ + hero_from_widget.tag + ], + self.duration, + ) + Animation( + pos=self.screen_in.to_widget( + *hero_from_widget.to_window( + *hero_from_widget.pos + ) + ), + size=hero_from_widget.size, + d=self.duration, + ).start( + self._hero_from_widget_children[ + hero_from_widget.tag + ] + ) def on_complete(self) -> None: + """ + Override method. + See :attr:`kivy.uix.screenmanager.TransitionBase.on_complete'. + """ + super().on_complete() + if self.manager._heroes_data and self.manager.current_heroes: + for hero_from_widget in self.manager.get_hero_from_widget(): + for heroes_tag in self.manager.current_heroes: + if heroes_tag == hero_from_widget.tag: + hero_from_children = self._hero_from_widget_children[ + heroes_tag + ] + self.manager.get_root_window().remove_widget( + hero_from_children + ) + + # Adding a child widget from the 'MDHeraFrom' container + # to the 'MDHeroTo' container. + if self._direction == "in": + for hero_to_widget in self.screen_in.heroes_to: + if hero_to_widget.tag == heroes_tag: + hero_to_widget.add_widget( + hero_from_children + ) + # Restores the child widget for the 'MDHeraFrom' + # container. + elif self._direction == "out": + hero_from_widget.add_widget(hero_from_children) + if self._direction == "out": self._direction = "in" - if self.manager._heroes_data and self.manager.current_hero: - self.manager.get_root_window().remove_widget(self.hero_widget) - self.hero_from_widget.add_widget(self.hero_widget) else: self._direction = "out" - if self.manager._heroes_data and self.manager.current_hero: - self.manager.get_root_window().remove_widget(self.hero_widget) - self.screen_in.hero_to.add_widget(self.hero_widget) - def _check_widget_properties(self): - if not self.screen_in.hero_to: + # Checks the attributes for the 'self.screen_in' screen. + # Called from the animated_hero_in method. + def _check_widget_properties(self, hero_from_widget: MDHeroFrom): + if not self.screen_in.heroes_to: raise Exception( - f"The `hero_to` attribute is not specified for screen {self.screen_in}" + f"The `heroes_to` attribute is not specified for screen " + f"{self.screen_in}" ) - if len(self.hero_from_widget.children) > 1: + # The 'MDHeroFrom' widget allows you to place only one widget in + # itself. + if len(hero_from_widget.children) > 1: raise Exception( - f"{self.hero_from_widget.__class__} accept only one widget" + f"{hero_from_widget.__class__} accept only one widget" ) + # For new API support. + def _check_hero_to_widget_tag( + self, hero_to_widget: MDHeroTo, hero_from_widget: MDHeroFrom + ) -> None: + if not hero_to_widget.tag: + Logger.warning( + "KivyMD: " + f"Set the tag '{hero_from_widget.tag}' " + f"for the {hero_to_widget} widget to the same " + f"as for the {hero_from_widget} widget" + ) + hero_to_widget.tag = hero_from_widget.tag + class MDSwapTransition(SwapTransition, MDTransitionBase): pass diff --git a/sbapp/kivymd/uix/widget.py b/sbapp/kivymd/uix/widget.py index 19a5d60..044722b 100644 --- a/sbapp/kivymd/uix/widget.py +++ b/sbapp/kivymd/uix/widget.py @@ -36,11 +36,13 @@ MDWidget __all__ = ("MDWidget",) +from kivy.uix.widget import Widget + from kivymd.uix import MDAdaptiveWidget from kivymd.uix.behaviors import DeclarativeBehavior -class MDWidget(DeclarativeBehavior, MDAdaptiveWidget): +class MDWidget(DeclarativeBehavior, MDAdaptiveWidget, Widget): """ See :class:`~kivy.uix.Widget` class documentation for more information.