mirror of
https://github.com/markqvist/Sideband.git
synced 2025-01-15 01:17:25 -05:00
1245 lines
34 KiB
Python
Executable File
1245 lines
34 KiB
Python
Executable File
"""
|
||
Components/Chip
|
||
===============
|
||
|
||
.. seealso::
|
||
|
||
`Material Design spec, Chips <https://m3.material.io/components/chips/overview>`_
|
||
|
||
.. rubric:: Chips can show multiple interactive elements together in the same
|
||
area, such as a list of selectable movie times, or a series of email
|
||
contacts. There are four types of chips: assist, filter, input, and
|
||
suggestion.
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/chips.png
|
||
:align: center
|
||
|
||
Usage
|
||
-----
|
||
|
||
.. tabs::
|
||
|
||
.. tab:: Declarative KV style
|
||
|
||
.. code-block:: python
|
||
|
||
from kivy.lang import Builder
|
||
|
||
from kivymd.app import MDApp
|
||
|
||
KV = '''
|
||
MDScreen:
|
||
|
||
MDChip:
|
||
pos_hint: {"center_x": .5, "center_y": .5}
|
||
|
||
MDChipText:
|
||
text: "MDChip"
|
||
'''
|
||
|
||
|
||
class Example(MDApp):
|
||
def build(self):
|
||
self.theme_cls.theme_style = "Dark"
|
||
return Builder.load_string(KV)
|
||
|
||
|
||
Example().run()
|
||
|
||
.. tab:: Declarative Python style
|
||
|
||
.. code-block:: python
|
||
|
||
from kivymd.app import MDApp
|
||
from kivymd.uix.chip import MDChip, MDChipText
|
||
from kivymd.uix.screen import MDScreen
|
||
|
||
|
||
class Example(MDApp):
|
||
def build(self):
|
||
self.theme_cls.theme_style = "Dark"
|
||
return (
|
||
MDScreen(
|
||
MDChip(
|
||
MDChipText(
|
||
text="MDChip"
|
||
),
|
||
pos_hint={"center_x": .5, "center_y": .5},
|
||
)
|
||
)
|
||
)
|
||
|
||
|
||
Example().run()
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/chip.png
|
||
:align: center
|
||
|
||
Anatomy
|
||
-------
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/anatomy-chip.png
|
||
:align: center
|
||
|
||
1. Container
|
||
2. Label text
|
||
3. Leading icon or image (optional)
|
||
4. Trailing remove icon (optional, input & filter chips only)
|
||
|
||
Container
|
||
---------
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/radius-chip.png
|
||
:align: center
|
||
|
||
All chips are slightly rounded with an 8dp corner.
|
||
|
||
Shadows and elevation
|
||
---------------------
|
||
|
||
Chip containers can be elevated if the placement requires protection, such as
|
||
on top of an image.
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/shadows-elevation-chip.png
|
||
:align: center
|
||
|
||
The following types of chips are available:
|
||
-------------------------------------------
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/available-type-chips.png
|
||
:align: center
|
||
|
||
- Assist_
|
||
- Filter_
|
||
- Input_
|
||
- Suggestion_
|
||
|
||
.. Assist:
|
||
Assist
|
||
------
|
||
|
||
`Assist chips <https://m3.material.io/components/chips/guidelines#5dd1928c-1476-4029-bdc5-fde66fc0dcb1>`_
|
||
represent smart or automated actions that can span multiple apps, such as
|
||
opening a calendar event from the home screen. Assist chips function as
|
||
though the user asked an assistant to complete the action. They should appear
|
||
dynamically and contextually in a UI.
|
||
|
||
An alternative to assist chips are buttons, which should appear persistently
|
||
and consistently.
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/assist-chip.png
|
||
:align: center
|
||
|
||
Example of assist
|
||
-----------------
|
||
|
||
.. code-block:: python
|
||
|
||
from kivy.lang import Builder
|
||
|
||
from kivymd.app import MDApp
|
||
|
||
KV = '''
|
||
<CommonLabel@MDLabel>
|
||
adaptive_size: True
|
||
theme_text_color: "Custom"
|
||
text_color: "#e6e9df"
|
||
|
||
|
||
<CommonAssistChip@MDChip>
|
||
# Custom attribute.
|
||
text: ""
|
||
icon: ""
|
||
|
||
# Chip attribute.
|
||
type: "assist"
|
||
md_bg_color: "#2a3127"
|
||
line_color: "grey"
|
||
elevation: 1
|
||
shadow_softness: 2
|
||
|
||
MDChipLeadingIcon:
|
||
icon: root.icon
|
||
theme_text_color: "Custom"
|
||
text_color: "#68896c"
|
||
|
||
MDChipText:
|
||
text: root.text
|
||
theme_text_color: "Custom"
|
||
text_color: "#e6e9df"
|
||
|
||
|
||
MDScreen:
|
||
|
||
FitImage:
|
||
source: "bg.png"
|
||
|
||
MDBoxLayout:
|
||
orientation: "vertical"
|
||
adaptive_size: True
|
||
pos_hint: {"center_y": .6, "center_x": .5}
|
||
|
||
CommonLabel:
|
||
text: "in 10 mins"
|
||
bold: True
|
||
pos_hint: {"center_x": .5}
|
||
|
||
CommonLabel:
|
||
text: "Therapy with Thea"
|
||
font_style: "H3"
|
||
padding_y: "12dp"
|
||
|
||
CommonLabel:
|
||
text: "Video call"
|
||
font_style: "H5"
|
||
pos_hint: {"center_x": .5}
|
||
|
||
MDBoxLayout:
|
||
adaptive_size: True
|
||
pos_hint: {"center_x": .5}
|
||
spacing: "12dp"
|
||
padding: 0, "24dp", 0, 0
|
||
|
||
CommonAssistChip:
|
||
text: "Home office"
|
||
icon: "map-marker"
|
||
|
||
CommonAssistChip:
|
||
text: "Chat"
|
||
icon: "message"
|
||
|
||
MDWidget:
|
||
'''
|
||
|
||
|
||
class Example(MDApp):
|
||
def build(self):
|
||
self.theme_cls.primary_palette = "Teal"
|
||
self.theme_cls.theme_style = "Dark"
|
||
return Builder.load_string(KV)
|
||
|
||
|
||
Example().run()
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/example-assist-chip.png
|
||
:align: center
|
||
|
||
.. Filter:
|
||
Filter
|
||
------
|
||
|
||
`Filter chips <https://m3.material.io/components/chips/guidelines#8d453d50-8d8e-43aa-9ae3-87ed134d2e64>`_
|
||
use tags or descriptive words to filter content. They can be a good alternative
|
||
to toggle buttons or checkboxes.
|
||
|
||
Tapping on a filter chip activates it and appends a leading checkmark icon to
|
||
the starting edge of the chip label.
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/filter-chip.png
|
||
:align: center
|
||
|
||
Example of filtering
|
||
--------------------
|
||
|
||
.. code-block:: python
|
||
|
||
from kivy.lang import Builder
|
||
from kivy.properties import StringProperty, ListProperty
|
||
|
||
from kivymd.app import MDApp
|
||
from kivymd.uix.chip import MDChip, MDChipText
|
||
from kivymd.uix.list import OneLineIconListItem
|
||
from kivymd.icon_definitions import md_icons
|
||
from kivymd.uix.screen import MDScreen
|
||
from kivymd.utils import asynckivy
|
||
|
||
Builder.load_string(
|
||
'''
|
||
<CustomOneLineIconListItem>
|
||
|
||
IconLeftWidget:
|
||
icon: root.icon
|
||
|
||
|
||
<PreviewIconsScreen>
|
||
|
||
MDBoxLayout:
|
||
orientation: "vertical"
|
||
spacing: "14dp"
|
||
padding: "20dp"
|
||
|
||
MDTextField:
|
||
id: search_field
|
||
hint_text: "Search icon"
|
||
mode: "rectangle"
|
||
icon_left: "magnify"
|
||
on_text: root.set_list_md_icons(self.text, True)
|
||
|
||
MDBoxLayout:
|
||
id: chip_box
|
||
spacing: "12dp"
|
||
adaptive_height: True
|
||
|
||
RecycleView:
|
||
id: rv
|
||
viewclass: "CustomOneLineIconListItem"
|
||
key_size: "height"
|
||
|
||
RecycleBoxLayout:
|
||
padding: dp(10)
|
||
default_size: None, dp(48)
|
||
default_size_hint: 1, None
|
||
size_hint_y: None
|
||
height: self.minimum_height
|
||
orientation: "vertical"
|
||
'''
|
||
)
|
||
|
||
|
||
class CustomOneLineIconListItem(OneLineIconListItem):
|
||
icon = StringProperty()
|
||
|
||
|
||
class PreviewIconsScreen(MDScreen):
|
||
filter = ListProperty() # list of tags for filtering icons
|
||
|
||
def set_filter_chips(self):
|
||
'''Asynchronously creates and adds chips to the container.'''
|
||
|
||
async def set_filter_chips():
|
||
for tag in ["Outline", "Off", "On"]:
|
||
await asynckivy.sleep(0)
|
||
chip = MDChip(
|
||
MDChipText(
|
||
text=tag,
|
||
),
|
||
type="filter",
|
||
md_bg_color="#303A29",
|
||
)
|
||
chip.bind(active=lambda x, y, z=tag: self.set_filter(y, z))
|
||
self.ids.chip_box.add_widget(chip)
|
||
|
||
asynckivy.start(set_filter_chips())
|
||
|
||
def set_filter(self, active: bool, tag: str) -> None:
|
||
'''Sets a list of tags for filtering icons.'''
|
||
|
||
if active:
|
||
self.filter.append(tag)
|
||
else:
|
||
self.filter.remove(tag)
|
||
|
||
def set_list_md_icons(self, text="", search=False) -> None:
|
||
'''Builds a list of icons.'''
|
||
|
||
def add_icon_item(name_icon):
|
||
self.ids.rv.data.append(
|
||
{
|
||
"icon": name_icon,
|
||
"text": name_icon,
|
||
}
|
||
)
|
||
|
||
self.ids.rv.data = []
|
||
for name_icon in md_icons.keys():
|
||
for tag in self.filter:
|
||
if tag.lower() in name_icon:
|
||
if search:
|
||
if text in name_icon:
|
||
add_icon_item(name_icon)
|
||
else:
|
||
add_icon_item(name_icon)
|
||
|
||
|
||
class Example(MDApp):
|
||
def __init__(self, **kwargs):
|
||
super().__init__(**kwargs)
|
||
self.screen = PreviewIconsScreen()
|
||
|
||
def build(self) -> PreviewIconsScreen:
|
||
self.theme_cls.theme_style = "Dark"
|
||
self.theme_cls.primary_palette = "LightGreen"
|
||
return self.screen
|
||
|
||
def on_start(self) -> None:
|
||
self.screen.set_list_md_icons()
|
||
self.screen.set_filter_chips()
|
||
|
||
|
||
Example().run()
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/example-filtering-icons-chip.gif
|
||
:align: center
|
||
|
||
Tap a chip to select it. Multiple chips can be selected or unselected:
|
||
|
||
.. code-block:: python
|
||
|
||
from kivy.lang import Builder
|
||
|
||
from kivymd.app import MDApp
|
||
from kivymd.uix.chip import MDChip, MDChipText
|
||
from kivymd.uix.screen import MDScreen
|
||
from kivymd.utils import asynckivy
|
||
|
||
Builder.load_string(
|
||
'''
|
||
<ChipScreen>
|
||
|
||
MDBoxLayout:
|
||
orientation: "vertical"
|
||
spacing: "14dp"
|
||
padding: "20dp"
|
||
|
||
MDLabel:
|
||
adaptive_height: True
|
||
text: "Select Type"
|
||
|
||
MDStackLayout:
|
||
id: chip_box
|
||
spacing: "12dp"
|
||
adaptive_height: True
|
||
|
||
MDWidget:
|
||
|
||
MDFlatButton:
|
||
text: "Uncheck chips"
|
||
pos: "20dp", "20dp"
|
||
on_release: root.unchecks_chips()
|
||
'''
|
||
)
|
||
|
||
|
||
class ChipScreen(MDScreen):
|
||
async def create_chips(self):
|
||
'''Asynchronously creates and adds chips to the container.'''
|
||
|
||
for tag in ["Extra Soft", "Soft", "Medium", "Hard"]:
|
||
await asynckivy.sleep(0)
|
||
self.ids.chip_box.add_widget(
|
||
MDChip(
|
||
MDChipText(
|
||
text=tag,
|
||
),
|
||
type="filter",
|
||
md_bg_color="#303A29",
|
||
active=True,
|
||
)
|
||
)
|
||
|
||
def unchecks_chips(self) -> None:
|
||
'''Removes marks from all chips.'''
|
||
|
||
for chip in self.ids.chip_box.children:
|
||
if chip.active:
|
||
chip.active = False
|
||
|
||
|
||
class Example(MDApp):
|
||
def __init__(self, **kwargs):
|
||
super().__init__(**kwargs)
|
||
self.screen = ChipScreen()
|
||
|
||
def build(self) -> ChipScreen:
|
||
self.theme_cls.theme_style = "Dark"
|
||
self.theme_cls.primary_palette = "LightGreen"
|
||
return self.screen
|
||
|
||
def on_start(self) -> None:
|
||
asynckivy.start(self.screen.create_chips())
|
||
|
||
|
||
Example().run()
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/example-filtering-icons-chip-2.gif
|
||
:align: center
|
||
|
||
Alternatively, a single chip can be selected.
|
||
This offers an alternative to toggle buttons, radio buttons, or single select
|
||
menus:
|
||
|
||
.. code-block:: python
|
||
|
||
from kivy.lang import Builder
|
||
|
||
from kivymd.app import MDApp
|
||
from kivymd.uix.chip import MDChip, MDChipText
|
||
from kivymd.uix.screen import MDScreen
|
||
from kivymd.utils import asynckivy
|
||
|
||
Builder.load_string(
|
||
'''
|
||
<ChipScreen>
|
||
|
||
MDBoxLayout:
|
||
orientation: "vertical"
|
||
spacing: "14dp"
|
||
padding: "20dp"
|
||
|
||
MDLabel:
|
||
adaptive_height: True
|
||
text: "Select Type"
|
||
|
||
MDStackLayout:
|
||
id: chip_box
|
||
spacing: "12dp"
|
||
adaptive_height: True
|
||
|
||
MDFillRoundFlatButton:
|
||
text: "Add to cart"
|
||
md_bg_color: "green"
|
||
size_hint_x: 1
|
||
|
||
MDWidget:
|
||
'''
|
||
)
|
||
|
||
|
||
class ChipScreen(MDScreen):
|
||
async def create_chips(self):
|
||
'''Asynchronously creates and adds chips to the container.'''
|
||
|
||
for tag in ["Extra Soft", "Soft", "Medium", "Hard"]:
|
||
await asynckivy.sleep(0)
|
||
chip = MDChip(
|
||
MDChipText(
|
||
text=tag,
|
||
),
|
||
type="filter",
|
||
md_bg_color="#303A29",
|
||
|
||
)
|
||
chip.bind(active=self.uncheck_chip)
|
||
self.ids.chip_box.add_widget(chip)
|
||
|
||
def uncheck_chip(self, current_chip: MDChip, active: bool) -> None:
|
||
'''Removes a mark from an already marked chip.'''
|
||
|
||
if active:
|
||
for chip in self.ids.chip_box.children:
|
||
if current_chip is not chip:
|
||
if chip.active:
|
||
chip.active = False
|
||
|
||
|
||
class Example(MDApp):
|
||
def __init__(self, **kwargs):
|
||
super().__init__(**kwargs)
|
||
self.screen = ChipScreen()
|
||
|
||
def build(self) -> ChipScreen:
|
||
self.theme_cls.theme_style = "Dark"
|
||
self.theme_cls.primary_palette = "LightGreen"
|
||
return self.screen
|
||
|
||
def on_start(self) -> None:
|
||
asynckivy.start(self.screen.create_chips())
|
||
|
||
|
||
Example().run()
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/example-filtering-single-select.gif
|
||
:align: center
|
||
|
||
.. Input:
|
||
Input
|
||
-----
|
||
|
||
`Input chips <https://m3.material.io/components/chips/guidelines#4d2d5ef5-3fcd-46e9-99f2-067747b2393f>`_
|
||
represent discrete pieces of information entered by a user, such as Gmail
|
||
contacts or filter options within a search field.
|
||
|
||
They enable user input and verify that input by converting text into chips.
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/input-chip.png
|
||
:align: center
|
||
|
||
Example of input
|
||
----------------
|
||
|
||
.. code-block:: python
|
||
|
||
from kivy.lang import Builder
|
||
|
||
from kivymd.app import MDApp
|
||
|
||
KV = '''
|
||
MDScreen:
|
||
|
||
MDChip:
|
||
pos_hint: {"center_x": .5, "center_y": .5}
|
||
type: "input"
|
||
line_color: "grey"
|
||
_no_ripple_effect: True
|
||
|
||
MDChipLeadingAvatar:
|
||
source: "data/logo/kivy-icon-128.png"
|
||
|
||
MDChipText:
|
||
text: "MDChip"
|
||
|
||
MDChipTrailingIcon:
|
||
icon: "close"
|
||
'''
|
||
|
||
|
||
class Example(MDApp):
|
||
def build(self):
|
||
self.theme_cls.theme_style = "Dark"
|
||
return Builder.load_string(KV)
|
||
|
||
|
||
Example().run()
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/example-input-chip.png
|
||
:align: center
|
||
|
||
.. Suggestion:
|
||
Suggestion
|
||
----------
|
||
|
||
`Suggestion chips <https://m3.material.io/components/chips/guidelines#36d7bb16-a9bf-4cf6-a73d-8e05510d66a7>`_
|
||
help narrow a user’s intent by presenting dynamically generated suggestions,
|
||
such as possible responses or search filters.
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/suggestion-chip.png
|
||
:align: center
|
||
|
||
Example of suggestion
|
||
---------------------
|
||
|
||
.. code-block::
|
||
|
||
from kivy.lang import Builder
|
||
|
||
from kivymd.app import MDApp
|
||
|
||
KV = '''
|
||
MDScreen:
|
||
|
||
MDChip:
|
||
pos_hint: {"center_x": .5, "center_y": .5}
|
||
type: "suggestion"
|
||
line_color: "grey"
|
||
|
||
MDChipText:
|
||
text: "MDChip"
|
||
'''
|
||
|
||
|
||
class Example(MDApp):
|
||
def build(self):
|
||
self.theme_cls.theme_style = "Dark"
|
||
return Builder.load_string(KV)
|
||
|
||
|
||
Example().run()
|
||
|
||
.. image:: https://github.com/HeaTTheatR/KivyMD-data/raw/master/gallery/kivymddoc/example-suggestion.png
|
||
:align: center
|
||
|
||
API break
|
||
=========
|
||
|
||
1.1.1 version
|
||
-------------
|
||
|
||
.. code-block:: python
|
||
|
||
from kivy.lang import Builder
|
||
|
||
from kivymd.app import MDApp
|
||
|
||
KV = '''
|
||
MDScreen:
|
||
|
||
MDChip:
|
||
text: "Portland"
|
||
pos_hint: {"center_x": .5, "center_y": .5}
|
||
on_release: app.on_release_chip(self)
|
||
'''
|
||
|
||
|
||
class Test(MDApp):
|
||
def build(self):
|
||
return Builder.load_string(KV)
|
||
|
||
def on_release_chip(self, instance_check):
|
||
print(instance_check)
|
||
|
||
|
||
Test().run()
|
||
|
||
1.2.0 version
|
||
-------------
|
||
|
||
.. code-block:: python
|
||
|
||
from kivy.lang import Builder
|
||
|
||
from kivymd.app import MDApp
|
||
|
||
KV = '''
|
||
MDScreen:
|
||
|
||
MDChip:
|
||
pos_hint: {"center_x": .5, "center_y": .5}
|
||
line_color: "grey"
|
||
on_release: app.on_release_chip(self)
|
||
|
||
MDChipText:
|
||
text: "MDChip"
|
||
'''
|
||
|
||
|
||
class Example(MDApp):
|
||
def build(self):
|
||
return Builder.load_string(KV)
|
||
|
||
def on_release_chip(self, instance_check):
|
||
print(instance_check)
|
||
|
||
|
||
Example().run()
|
||
"""
|
||
|
||
from __future__ import annotations
|
||
|
||
__all__ = (
|
||
"MDChip",
|
||
"MDChipLeadingAvatar",
|
||
"MDChipLeadingIcon",
|
||
"MDChipTrailingIcon",
|
||
"MDChipText",
|
||
)
|
||
|
||
import os
|
||
|
||
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 (
|
||
BooleanProperty,
|
||
ColorProperty,
|
||
OptionProperty,
|
||
StringProperty,
|
||
VariableListProperty,
|
||
)
|
||
from kivy.uix.behaviors import ButtonBehavior
|
||
|
||
from kivymd import uix_path
|
||
from kivymd.material_resources import DEVICE_TYPE
|
||
from kivymd.uix.behaviors import (
|
||
CircularRippleBehavior,
|
||
CommonElevationBehavior,
|
||
RectangularRippleBehavior,
|
||
ScaleBehavior,
|
||
TouchBehavior,
|
||
)
|
||
from kivymd.uix.boxlayout import MDBoxLayout
|
||
from kivymd.uix.label import MDIcon, MDLabel
|
||
|
||
with open(
|
||
os.path.join(uix_path, "chip", "chip.kv"), encoding="utf-8"
|
||
) as kv_file:
|
||
Builder.load_string(kv_file.read())
|
||
|
||
|
||
class BaseChipIcon(
|
||
CircularRippleBehavior, ScaleBehavior, ButtonBehavior, MDIcon
|
||
):
|
||
def __init__(self, **kwargs):
|
||
super().__init__(**kwargs)
|
||
self.ripple_scale = 1.5
|
||
Clock.schedule_once(self.adjust_icon_size)
|
||
|
||
def adjust_icon_size(self, *args) -> None:
|
||
# If the user has not changed the icon size, then we set the standard
|
||
# icon size according to the standards of material design version 3.
|
||
if (
|
||
self.font_name == "Icons"
|
||
and self.theme_cls.font_styles["Icon"][1] == self.font_size
|
||
):
|
||
self.font_size = (
|
||
"18sp"
|
||
if not self.source and not isinstance(self, MDChipLeadingAvatar)
|
||
else "24sp"
|
||
)
|
||
if self.source and isinstance(self, MDChipLeadingAvatar):
|
||
self.icon = self.source
|
||
self._size = [dp(28), dp(28)]
|
||
self.font_size = "28sp"
|
||
self.padding_x = "6dp"
|
||
self._no_ripple_effect = True
|
||
|
||
|
||
class LabelTextContainer(MDBoxLayout):
|
||
"""Implements a container for the chip label."""
|
||
|
||
|
||
class LeadingIconContainer(MDBoxLayout):
|
||
"""Implements a container for the leading icon."""
|
||
|
||
|
||
class TrailingIconContainer(MDBoxLayout):
|
||
"""Implements a container for the trailing icon."""
|
||
|
||
|
||
class MDChipLeadingAvatar(BaseChipIcon):
|
||
"""
|
||
Implements the leading avatar for the chip.
|
||
|
||
For more information, see in the
|
||
:class:`~kivymd.uix.behaviors.CircularRippleBehavior` and
|
||
:class:`~kivymd.uix.behaviors.ScaleBehavior` and
|
||
:class:`~kivy.uix.behaviors.ButtonBehavior` and
|
||
:class:`~kivymd.uix.label.MDIcon`
|
||
classes documentation.
|
||
"""
|
||
|
||
|
||
class MDChipLeadingIcon(BaseChipIcon):
|
||
"""
|
||
Implements the leading icon for the chip.
|
||
|
||
For more information, see in the
|
||
:class:`~kivymd.uix.behaviors.CircularRippleBehavior` and
|
||
:class:`~kivymd.uix.behaviors.ScaleBehavior` and
|
||
:class:`~kivy.uix.behaviors.ButtonBehavior` and
|
||
:class:`~kivymd.uix.label.MDIcon`
|
||
classes documentation.
|
||
"""
|
||
|
||
|
||
class MDChipTrailingIcon(BaseChipIcon):
|
||
"""
|
||
Implements the trailing icon for the chip.
|
||
|
||
For more information, see in the
|
||
:class:`~kivymd.uix.behaviors.CircularRippleBehavior` and
|
||
:class:`~kivymd.uix.behaviors.ScaleBehavior` and
|
||
:class:`~kivy.uix.behaviors.ButtonBehavior` and
|
||
:class:`~kivymd.uix.label.MDIcon`
|
||
classes documentation.
|
||
"""
|
||
|
||
|
||
class MDChipText(MDLabel):
|
||
"""
|
||
Implements the label for the chip.
|
||
|
||
For more information, see in the
|
||
:class:`~kivymd.uix.label.MDLabel` classes documentation.
|
||
"""
|
||
|
||
|
||
class MDChip(
|
||
MDBoxLayout,
|
||
RectangularRippleBehavior,
|
||
ButtonBehavior,
|
||
CommonElevationBehavior,
|
||
TouchBehavior,
|
||
):
|
||
"""
|
||
Chip class.
|
||
|
||
For more information, see in the
|
||
:class:`~kivymd.uix.boxlayout.MDBoxLayout` and
|
||
:class:`~kivymd.uix.behaviors.RectangularRippleBehavior` and
|
||
:class:`~kivy.uix.behaviors.ButtonBehavior` and
|
||
:class:`~kivymd.uix.behaviors.CommonElevationBehavior` and
|
||
:class:`~kivymd.uix.behaviors.TouchBehavior`
|
||
classes documentation.
|
||
"""
|
||
|
||
radius = VariableListProperty([dp(8)], length=4)
|
||
"""
|
||
Chip radius.
|
||
|
||
:attr:`radius` is an :class:`~kivy.properties.VariableListProperty`
|
||
and defaults to `[dp(8), dp(8), dp(8), dp(8)]`.
|
||
"""
|
||
|
||
text = StringProperty(deprecated=True)
|
||
"""
|
||
Chip text.
|
||
|
||
.. deprecated:: 1.2.0
|
||
|
||
:attr:`text` is an :class:`~kivy.properties.StringProperty`
|
||
and defaults to `''`.
|
||
"""
|
||
|
||
type = OptionProperty(
|
||
"suggestion", options=["assist", "filter", "input", "suggestion"]
|
||
)
|
||
"""
|
||
Type of chip.
|
||
|
||
.. versionadded:: 1.2.0
|
||
|
||
Available options are: `'assist'`, `'filter'`, `'input'`, `'suggestion'`.
|
||
|
||
:attr:`type` is an :class:`~kivy.properties.OptionProperty`
|
||
and defaults to `'suggestion'`.
|
||
"""
|
||
|
||
icon_left = StringProperty(deprecated=True)
|
||
"""
|
||
Chip left icon.
|
||
|
||
.. versionadded:: 1.0.0
|
||
|
||
.. deprecated:: 1.2.0
|
||
|
||
:attr:`icon_left` is an :class:`~kivy.properties.StringProperty`
|
||
and defaults to `''`.
|
||
"""
|
||
|
||
icon_right = StringProperty(deprecated=True)
|
||
"""
|
||
Chip right icon.
|
||
|
||
.. versionadded:: 1.0.0
|
||
|
||
.. deprecated:: 1.2.0
|
||
|
||
:attr:`icon_right` is an :class:`~kivy.properties.StringProperty`
|
||
and defaults to `''`.
|
||
"""
|
||
|
||
text_color = ColorProperty(None, deprecated=True)
|
||
"""
|
||
Chip's text color in (r, g, b, a) or string format.
|
||
|
||
.. deprecated:: 1.2.0
|
||
|
||
:attr:`text_color` is an :class:`~kivy.properties.ColorProperty`
|
||
and defaults to `None`.
|
||
"""
|
||
|
||
icon_right_color = ColorProperty(None, deprecated=True)
|
||
"""
|
||
Chip's right icon color in (r, g, b, a) or string format.
|
||
|
||
.. versionadded:: 1.0.0
|
||
|
||
.. deprecated:: 1.2.0
|
||
|
||
:attr:`icon_right_color` is an :class:`~kivy.properties.ColorProperty`
|
||
and defaults to `None`.
|
||
"""
|
||
|
||
icon_left_color = ColorProperty(None, deprecated=True)
|
||
"""
|
||
Chip's left icon color in (r, g, b, a) or string format.
|
||
|
||
.. versionadded:: 1.0.0
|
||
|
||
.. deprecated:: 1.2.0
|
||
|
||
:attr:`icon_left_color` is an :class:`~kivy.properties.ColorProperty`
|
||
and defaults to `None`.
|
||
"""
|
||
|
||
icon_check_color = ColorProperty(None)
|
||
"""
|
||
Chip's check icon color in (r, g, b, a) or string format.
|
||
|
||
.. versionadded:: 1.0.0
|
||
|
||
:attr:`icon_check_color` is an :class:`~kivy.properties.ColorProperty`
|
||
and defaults to `None`.
|
||
"""
|
||
|
||
active = BooleanProperty(False)
|
||
"""
|
||
Whether the check is marked or not.
|
||
|
||
.. versionadded:: 1.0.0
|
||
|
||
:attr:`active` is an :class:`~kivy.properties.BooleanProperty`
|
||
and defaults to `False`.
|
||
"""
|
||
|
||
selected_color = ColorProperty(None)
|
||
"""
|
||
The background color of the chip in the marked state in (r, g, b, a)
|
||
or string format.
|
||
|
||
.. versionadded:: 1.2.0
|
||
|
||
:attr:`selected_color` is an :class:`~kivy.properties.ColorProperty`
|
||
and defaults to `None`.
|
||
"""
|
||
|
||
_current_md_bg_color = ColorProperty(None)
|
||
# A flag that disallow ripple animation of the chip
|
||
# at the time of clicking the chip icons.
|
||
_allow_chip_ripple = BooleanProperty(True)
|
||
# The flag signals the end of the ripple animation.
|
||
_anim_complete = BooleanProperty(False)
|
||
|
||
def __init__(self, *args, **kwargs):
|
||
super().__init__(*args, **kwargs)
|
||
|
||
def on_long_touch(self, *args) -> None:
|
||
if self.type == "filter":
|
||
self.active = not self.active
|
||
|
||
def on_type(self, instance, value: str) -> None:
|
||
"""Called when the values of :attr:`type` change."""
|
||
|
||
def adjust_padding(*args):
|
||
"""
|
||
According to the type of chip, it sets the margins according
|
||
to the specification of the material design version 3.
|
||
"""
|
||
|
||
self.padding = {
|
||
"input": (
|
||
"12dp"
|
||
if not self.ids.leading_icon_container.children
|
||
else (
|
||
"5dp"
|
||
if not self.ids.leading_icon_container.children[
|
||
0
|
||
].source
|
||
else "16dp"
|
||
),
|
||
0,
|
||
"4dp",
|
||
0,
|
||
),
|
||
"assist": (
|
||
"16dp"
|
||
if not self.ids.leading_icon_container.children
|
||
else "8dp",
|
||
0,
|
||
"16dp"
|
||
if not self.ids.leading_icon_container.children
|
||
else "8dp",
|
||
0,
|
||
),
|
||
"suggestion": (
|
||
"16dp"
|
||
if not self.ids.leading_icon_container.children
|
||
else "8dp",
|
||
0,
|
||
"16dp",
|
||
0,
|
||
),
|
||
"filter": (
|
||
"16dp"
|
||
if not self.ids.leading_icon_container.children
|
||
else (
|
||
"8dp"
|
||
if not self.ids.leading_icon_container.children[
|
||
0
|
||
].source
|
||
else "4dp"
|
||
),
|
||
0,
|
||
"16dp"
|
||
if not self.ids.trailing_icon_container.children
|
||
else "8dp",
|
||
0,
|
||
),
|
||
}[value]
|
||
|
||
Clock.schedule_once(adjust_padding)
|
||
|
||
def on_active(self, instance_check, active_value: bool) -> None:
|
||
"""Called when the values of :attr:`active` change."""
|
||
|
||
if active_value:
|
||
self._current_md_bg_color = self.md_bg_color
|
||
|
||
Clock.schedule_once(self.complete_anim_ripple, 0.5)
|
||
|
||
def complete_anim_ripple(self, *args) -> None:
|
||
"""Called at the end of the ripple animation."""
|
||
|
||
if self.active:
|
||
if not self.ids.leading_icon_container.children:
|
||
if self.type == "filter":
|
||
self.add_marked_icon_to_chip()
|
||
self.set_chip_bg_color(
|
||
self.selected_color
|
||
if self.selected_color
|
||
else self.theme_cls.primary_color
|
||
)
|
||
else:
|
||
if (
|
||
self.ids.leading_icon_container.children
|
||
and self.ids.leading_icon_container.children[0].icon == "check"
|
||
):
|
||
if self.type == "filter":
|
||
self.remove_marked_icon_from_chip()
|
||
self.set_chip_bg_color(self._current_md_bg_color)
|
||
|
||
def remove_marked_icon_from_chip(self) -> None:
|
||
def remove_marked_icon_from_chip(*args):
|
||
self.ids.leading_icon_container.clear_widgets()
|
||
|
||
if self.ids.leading_icon_container.children:
|
||
anim = Animation(scale_value_x=0, scale_value_y=0, d=0.2)
|
||
anim.bind(on_complete=remove_marked_icon_from_chip)
|
||
anim.start(self.ids.leading_icon_container.children[0])
|
||
Animation(
|
||
padding=[dp(16), 0, dp(16), 0],
|
||
spacing=0,
|
||
d=0.2,
|
||
).start(self)
|
||
|
||
def add_marked_icon_to_chip(self) -> None:
|
||
"""Adds and animates a check icon to the chip."""
|
||
|
||
icon_check = MDChipLeadingIcon(
|
||
icon="check",
|
||
pos_hint={"center_y": 0.5},
|
||
font_size=dp(18),
|
||
scale_value_x=0,
|
||
scale_value_y=0,
|
||
)
|
||
icon_check.bind(
|
||
on_press=self._set_allow_chip_ripple,
|
||
on_release=self._set_allow_chip_ripple,
|
||
)
|
||
self.ids.leading_icon_container.add_widget(icon_check)
|
||
# Animating the scale of the icon.
|
||
Animation(scale_value_x=1, scale_value_y=1, d=0.2).start(icon_check)
|
||
# Animating the padding of the chip.
|
||
Animation(
|
||
padding=[dp(18), 0, 0, 0],
|
||
spacing=dp(18) if self.type == "filter" else 0,
|
||
d=0.2,
|
||
).start(self)
|
||
|
||
def set_chip_bg_color(self, color: list | str) -> None:
|
||
"""Animates the background color of the chip."""
|
||
|
||
if color:
|
||
Animation(md_bg_color=color, d=0.2).start(self)
|
||
self._anim_complete = not self._anim_complete
|
||
|
||
def on_press(self, *args):
|
||
if self.active:
|
||
self.active = False
|
||
|
||
def add_widget(self, widget, *args, **kwargs):
|
||
def add_icon_leading_trailing(container):
|
||
if len(container.children):
|
||
type_icon = (
|
||
"'leading'"
|
||
if isinstance(
|
||
widget, (MDChipLeadingIcon, MDChipLeadingAvatar)
|
||
)
|
||
else "'trailing'"
|
||
)
|
||
Logger.warning(
|
||
f"KivyMD: "
|
||
f"Do not use more than one {type_icon} icon. "
|
||
f"This is contrary to the material design rules "
|
||
f"of version 3"
|
||
)
|
||
return
|
||
if isinstance(widget, MDChipTrailingIcon) and self.type in [
|
||
"assist",
|
||
"suggestion",
|
||
]:
|
||
Logger.warning(
|
||
f"KivyMD: "
|
||
f"According to the material design standards of version "
|
||
f"3, do not use the trailing icon for an '{self.type}' "
|
||
f"type chip."
|
||
)
|
||
return
|
||
if (
|
||
isinstance(widget, MDChipTrailingIcon)
|
||
and self.type == "filter"
|
||
and DEVICE_TYPE == "mobile"
|
||
):
|
||
Logger.warning(
|
||
"KivyMD: "
|
||
"According to the material design standards of version 3, "
|
||
"only on desktop computers and tablets, filter chips can "
|
||
"contain a finishing icon for directly removing the chip "
|
||
"or opening the options menu."
|
||
)
|
||
return
|
||
if (
|
||
isinstance(widget, (MDChipLeadingIcon, MDChipLeadingAvatar))
|
||
and self.type == "filter"
|
||
):
|
||
Logger.warning(
|
||
"KivyMD: "
|
||
"According to the material design standards of version 3, "
|
||
"it is better not to use a leading icon for a 'filter' "
|
||
"type chip."
|
||
)
|
||
if (
|
||
isinstance(widget, MDChipLeadingAvatar)
|
||
and self.type == "suggestion"
|
||
):
|
||
Logger.warning(
|
||
"KivyMD: "
|
||
"According to the material design standards of version 3, "
|
||
"it is better not to use a leading avatar for a "
|
||
"'suggestion' type chip."
|
||
)
|
||
return
|
||
|
||
widget.bind(
|
||
on_press=self._set_allow_chip_ripple,
|
||
on_release=self._set_allow_chip_ripple,
|
||
)
|
||
widget.pos_hint = {"center_y": 0.5}
|
||
self.padding = ("8dp", 0, "8dp", 0)
|
||
self.spacing = (
|
||
"8dp"
|
||
if isinstance(
|
||
widget,
|
||
(
|
||
MDChipLeadingIcon,
|
||
MDChipLeadingAvatar,
|
||
MDChipTrailingIcon,
|
||
),
|
||
)
|
||
else 0
|
||
)
|
||
container.add_widget(widget)
|
||
|
||
if isinstance(widget, MDChipText):
|
||
widget.adaptive_size = True
|
||
widget.pos_hint = {"center_y": 0.5}
|
||
if self.type == "suggestion":
|
||
self.padding = ("16dp", 0, "16dp", 0)
|
||
Clock.schedule_once(
|
||
lambda x: self.ids.label_container.add_widget(widget)
|
||
)
|
||
elif isinstance(widget, (MDChipLeadingIcon, MDChipLeadingAvatar)):
|
||
Clock.schedule_once(
|
||
lambda x: add_icon_leading_trailing(
|
||
self.ids.leading_icon_container
|
||
)
|
||
)
|
||
elif isinstance(widget, MDChipTrailingIcon):
|
||
Clock.schedule_once(
|
||
lambda x: add_icon_leading_trailing(
|
||
self.ids.trailing_icon_container
|
||
)
|
||
)
|
||
elif isinstance(
|
||
widget,
|
||
(LabelTextContainer, LeadingIconContainer, TrailingIconContainer),
|
||
):
|
||
return super().add_widget(widget)
|
||
|
||
def _set_allow_chip_ripple(
|
||
self, instance: MDChipLeadingIcon | MDChipTrailingIcon
|
||
) -> None:
|
||
self._allow_chip_ripple = not self._allow_chip_ripple
|