ChipGroup одиночный выбор
Как я могу заставить ChipGroup
вести себя как RadioGroup
как наличие хотя бы одного выбранного элемента всегда? настройка setSingleSelection(true)
также добавляет возможность не выбирать ничего, если дважды щелкнуть Chip
,
8 ответов
Решением может быть установка фишки, по которой щелкают, а затем переключение свойства фишек, по которой можно нажимать:
chipGroup.setOnCheckedChangeListener((chipGroup, id) -> {
Chip chip = ((Chip) chipGroup.getChildAt(chipGroup.getCheckedChipId()));
if (chip != null) {
for (int i = 0; i < chipGroup.getChildCount(); ++i) {
chipGroup.getChildAt(i).setClickable(true);
}
chip.setClickable(false);
}
});
Чтобы не допустить отмены выбора всех фишек, вы можете использовать метод setSelectionRequired
:
chipGroup.setSelectionRequired(true)
Вы также можете определить его в макете, используя app:selectionRequired
атрибут:
<com.google.android.material.chip.ChipGroup
app:singleSelection="true"
app:selectionRequired="true"
app:checkedChip="@id/..."
..>
Примечание. Для этого требуется как минимум версия 1.2.0-alpha02.
РЕДАКТИРОВАТЬ
С версией 1.2.0-alpha02
не старый Hacky решение больше не требуется!
Либо используйте атрибут app:selectionRequired="true"
<com.google.android.material.chip.ChipGroup
android:id="@+id/group"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:selectionRequired="true"
app:singleSelection="true">
(...)
</com.google.android.material.chip.ChipGroup>
Или в коде
// Kotlin
group.isSelectionRequired = true
// Java
group.setSelectionRequired(true);
Для более старых версий
Для этого есть два шага
Шаг 1
У нас есть встроенная поддержка, просто не забудьте добавить app:singleSelection="true"
на ваш ChipGroup
, например:
XML
<com.google.android.material.chip.ChipGroup
android:id="@+id/group"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:singleSelection="true">
<com.google.android.material.chip.Chip
android:id="@+id/option_1"
style="@style/Widget.MaterialComponents.Chip.Choice"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Option 1" />
<com.google.android.material.chip.Chip
android:id="@+id/option_2"
style="@style/Widget.MaterialComponents.Chip.Choice"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Option 2" />
</com.google.android.material.chip.ChipGroup>
Код
// Kotlin
group.isSingleSelection = true
// Java
group.setSingleSelection(true);
Шаг 2
Теперь для поддержки функциональности радиогруппы:
var lastCheckedId = View.NO_ID
chipGroup.setOnCheckedChangeListener { group, checkedId ->
if(checkedId == View.NO_ID) {
// User tried to uncheck, make sure to keep the chip checked
group.check(lastCheckedId)
return@setOnCheckedChangeListener
}
lastCheckedId = checkedId
// New selection happened, do your logic here.
(...)
}
Из документов:
ChipGroup также поддерживает область множественного исключения для набора микросхем. Когда вы устанавливаете атрибут app:singleSelection, проверка одного чипа, принадлежащего группе чипов, снимает отметку с любого ранее отмеченного чипа в той же группе. Поведение отражает поведение RadioGroup.
Краткая модификация ответа @adriennoir (на Kotlin). Спасибо за помощь! Обратите внимание, что getChildAt()
берет индекс.
for (i in 0 until group.childCount) {
val chip = group.getChildAt(i)
chip.isClickable = chip.id != group.checkedChipId
}
Вот мой большой `setOnCheckedChangeListener, для контекста:
intervalChipGroup.setOnCheckedChangeListener { group, checkedId ->
for (i in 0 until group.childCount) {
val chip = group.getChildAt(i)
chip.isClickable = chip.id != group.checkedChipId
}
when (checkedId) {
R.id.intervalWeek -> {
view.findViewById<Chip>(R.id.intervalWeek).chipStrokeWidth = 1F
view.findViewById<Chip>(R.id.intervalMonth).chipStrokeWidth = 0F
view.findViewById<Chip>(R.id.intervalYear).chipStrokeWidth = 0F
currentIntervalSelected = weekInterval
populateGraph(weekInterval)
}
R.id.intervalMonth -> {
view.findViewById<Chip>(R.id.intervalWeek).chipStrokeWidth = 0F
view.findViewById<Chip>(R.id.intervalMonth).chipStrokeWidth = 1F
view.findViewById<Chip>(R.id.intervalYear).chipStrokeWidth = 0F
currentIntervalSelected = monthInterval
populateGraph(monthInterval)
}
R.id.intervalYear -> {
view.findViewById<Chip>(R.id.intervalWeek).chipStrokeWidth = 0F
view.findViewById<Chip>(R.id.intervalMonth).chipStrokeWidth = 0F
view.findViewById<Chip>(R.id.intervalYear).chipStrokeWidth = 1F
currentIntervalSelected = yearInterval
populateGraph(yearInterval)
}
}
}
Большинство ответов великолепны и действительно полезны для меня. Еще одна небольшая модификация @adriennoir и @Todd DeLand, чтобы предотвратить снятие отметки с уже проверенного чипа в setSingleSelection(true)
ChipGroup, вот мое решение:
for (i in 0 until chipGroup.childCount) {
val chip = chipGroup.getChildAt(i) as Chip
chip.isCheckable = chip.id != chipGroup.checkedChipId
chip.isChecked = chip.id == chipGroup.checkedChipId
}
Для меня, мне просто нужно, чтобы тот же проверенный чип не был снят, чтобы он не был кликабельным. Таким образом, пользователь все еще может щелкнуть по проверенному чипу и увидеть причудливый эффект ряби, и ничего не произойдет.
Если singleSelection не работает с динамически добавляемыми чипами, вы должны сгенерировать идентификатор для каждого чипа при их создании, а затем добавить в ChipGroup .
val chip = inflater.inflate(
R.layout.item_crypto_currency_category_chip,
binding.chipGroupCryptoCurrencyCategory,
false) as Chip
chip.id = ViewCompat.generateViewId()
binding.chipGroupCryptoCurrencyCategory.addView(chip)
//Set default value with index 0 when ChipGroup created.
if (index == 0) binding.chipGroupCryptoCurrencyCategory.check(chip.id)
item_crypto_currency_category_chip.xml
<?xml version="1.0" encoding="utf-8"?>
<com.google.android.material.chip.Chip xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/chip_smart_contract"
style="@style/Widget.Signal.Chip"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
item_crypto_currency_tag_category.xml
<HorizontalScrollView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="@dimen/spacing_6x"
android:scrollbars="none"
app:layout_constraintTop_toTopOf="parent">
<com.google.android.material.chip.ChipGroup
android:id="@+id/chip_group_crypto_currency_category"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:singleSelection="true"
app:singleLine="true"
/>
</HorizontalScrollView>
Результат:
Вот как я это сделал:
var previousSelection: Int = default_selection_id
chipGroup.setOnCheckedChangeListener { chipGroup, id ->
if (id == -1) //nothing is selected.
chipGroup.check(previousSelection)
else
previousSelection = id
Это мое рабочее решение
mChipGroup.setOnCheckedChangeListener((group, checkedId) -> {
for (int i = 0; i < mChipGroup.getChildCount(); i++) {
Chip chip = (Chip) mChipGroup.getChildAt(i);
if (chip != null) {
chip.setClickable(!(chip.getId() == mChipGroup.getCheckedChipId()));
}
}
});