|
|
|
|
@ -2,12 +2,14 @@
|
|
|
|
|
|
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
|
|
|
|
|
|
import 'package:dropdown_button2/dropdown_button2.dart';
|
|
|
|
|
|
|
|
|
|
import 'package:fluffychat/config/themes.dart';
|
|
|
|
|
import 'package:fluffychat/pangea/learning_settings/enums/l2_support_enum.dart';
|
|
|
|
|
import 'package:fluffychat/pangea/learning_settings/models/language_model.dart';
|
|
|
|
|
import 'flag.dart';
|
|
|
|
|
|
|
|
|
|
class PLanguageDropdown extends StatelessWidget {
|
|
|
|
|
class PLanguageDropdown extends StatefulWidget {
|
|
|
|
|
final List<LanguageModel> languages;
|
|
|
|
|
final LanguageModel? initialLanguage;
|
|
|
|
|
final Function(LanguageModel) onChange;
|
|
|
|
|
@ -29,9 +31,22 @@ class PLanguageDropdown extends StatelessWidget {
|
|
|
|
|
this.validator,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
PLanguageDropdownState createState() => PLanguageDropdownState();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class PLanguageDropdownState extends State<PLanguageDropdown> {
|
|
|
|
|
final TextEditingController _searchController = TextEditingController();
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
void dispose() {
|
|
|
|
|
_searchController.dispose();
|
|
|
|
|
super.dispose();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
final List<LanguageModel> sortedLanguages = languages;
|
|
|
|
|
final List<LanguageModel> sortedLanguages = widget.languages;
|
|
|
|
|
final String systemLang = Localizations.localeOf(context).languageCode;
|
|
|
|
|
final List<String> languagePriority = [systemLang, 'en', 'es'];
|
|
|
|
|
|
|
|
|
|
@ -60,16 +75,26 @@ class PLanguageDropdown extends StatelessWidget {
|
|
|
|
|
return Column(
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
children: [
|
|
|
|
|
DropdownButtonFormField<LanguageModel>(
|
|
|
|
|
decoration: InputDecoration(labelText: decorationText),
|
|
|
|
|
DropdownButtonFormField2<LanguageModel>(
|
|
|
|
|
customButton: widget.initialLanguage != null
|
|
|
|
|
? LanguageDropDownEntry(
|
|
|
|
|
languageModel: widget.initialLanguage!,
|
|
|
|
|
isL2List: widget.isL2List,
|
|
|
|
|
isDropdown: true,
|
|
|
|
|
)
|
|
|
|
|
: null,
|
|
|
|
|
decoration: InputDecoration(labelText: widget.decorationText),
|
|
|
|
|
isExpanded: true,
|
|
|
|
|
dropdownStyleData: const DropdownStyleData(
|
|
|
|
|
maxHeight: 400,
|
|
|
|
|
),
|
|
|
|
|
items: [
|
|
|
|
|
if (showMultilingual)
|
|
|
|
|
if (widget.showMultilingual)
|
|
|
|
|
DropdownMenuItem(
|
|
|
|
|
value: LanguageModel.multiLingual(context),
|
|
|
|
|
child: LanguageDropDownEntry(
|
|
|
|
|
languageModel: LanguageModel.multiLingual(context),
|
|
|
|
|
isL2List: isL2List,
|
|
|
|
|
isL2List: widget.isL2List,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
...sortedLanguages.map(
|
|
|
|
|
@ -77,18 +102,42 @@ class PLanguageDropdown extends StatelessWidget {
|
|
|
|
|
value: languageModel,
|
|
|
|
|
child: LanguageDropDownEntry(
|
|
|
|
|
languageModel: languageModel,
|
|
|
|
|
isL2List: isL2List,
|
|
|
|
|
isL2List: widget.isL2List,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
onChanged: (value) => onChange(value!),
|
|
|
|
|
value: initialLanguage,
|
|
|
|
|
validator: (value) => validator?.call(value),
|
|
|
|
|
onChanged: (value) => widget.onChange(value!),
|
|
|
|
|
value: widget.initialLanguage,
|
|
|
|
|
validator: (value) => widget.validator?.call(value),
|
|
|
|
|
dropdownSearchData: DropdownSearchData(
|
|
|
|
|
searchController: _searchController,
|
|
|
|
|
searchInnerWidgetHeight: 50,
|
|
|
|
|
searchInnerWidget: Padding(
|
|
|
|
|
padding: const EdgeInsets.symmetric(horizontal: 14, vertical: 10),
|
|
|
|
|
child: TextField(
|
|
|
|
|
autofocus: true,
|
|
|
|
|
controller: _searchController,
|
|
|
|
|
decoration: const InputDecoration(
|
|
|
|
|
prefixIcon: Icon(Icons.search),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
searchMatchFn: (item, searchValue) {
|
|
|
|
|
final displayName = item.value?.displayName.toLowerCase();
|
|
|
|
|
if (displayName == null) return false;
|
|
|
|
|
|
|
|
|
|
final search = searchValue.toLowerCase();
|
|
|
|
|
return displayName.startsWith(search);
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
onMenuStateChange: (isOpen) {
|
|
|
|
|
if (!isOpen) _searchController.clear();
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
AnimatedSize(
|
|
|
|
|
duration: FluffyThemes.animationDuration,
|
|
|
|
|
child: error == null
|
|
|
|
|
child: widget.error == null
|
|
|
|
|
? const SizedBox.shrink()
|
|
|
|
|
: Padding(
|
|
|
|
|
padding: const EdgeInsets.symmetric(
|
|
|
|
|
@ -96,7 +145,7 @@ class PLanguageDropdown extends StatelessWidget {
|
|
|
|
|
vertical: 5,
|
|
|
|
|
),
|
|
|
|
|
child: Text(
|
|
|
|
|
error!,
|
|
|
|
|
widget.error!,
|
|
|
|
|
style: TextStyle(
|
|
|
|
|
color: Theme.of(context).colorScheme.error,
|
|
|
|
|
fontSize: 12,
|
|
|
|
|
@ -112,40 +161,46 @@ class PLanguageDropdown extends StatelessWidget {
|
|
|
|
|
class LanguageDropDownEntry extends StatelessWidget {
|
|
|
|
|
final LanguageModel languageModel;
|
|
|
|
|
final bool isL2List;
|
|
|
|
|
final bool isDropdown;
|
|
|
|
|
|
|
|
|
|
const LanguageDropDownEntry({
|
|
|
|
|
super.key,
|
|
|
|
|
required this.languageModel,
|
|
|
|
|
required this.isL2List,
|
|
|
|
|
this.isDropdown = false,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
return Padding(
|
|
|
|
|
padding: const EdgeInsets.only(left: 12),
|
|
|
|
|
child: Row(
|
|
|
|
|
mainAxisAlignment: MainAxisAlignment.start,
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
|
|
|
|
return Row(
|
|
|
|
|
children: [
|
|
|
|
|
LanguageFlag(
|
|
|
|
|
language: languageModel,
|
|
|
|
|
),
|
|
|
|
|
const SizedBox(width: 10),
|
|
|
|
|
Flexible(
|
|
|
|
|
child: Text(
|
|
|
|
|
Expanded(
|
|
|
|
|
child: Row(
|
|
|
|
|
children: [
|
|
|
|
|
Text(
|
|
|
|
|
languageModel.getDisplayName(context) ?? "",
|
|
|
|
|
style: const TextStyle().copyWith(
|
|
|
|
|
color: Theme.of(context).textTheme.bodyLarge!.color,
|
|
|
|
|
fontSize: 14,
|
|
|
|
|
),
|
|
|
|
|
overflow: TextOverflow.ellipsis,
|
|
|
|
|
textAlign: TextAlign.center,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
const SizedBox(width: 10),
|
|
|
|
|
if (isL2List && languageModel.l2Support != L2SupportEnum.full)
|
|
|
|
|
languageModel.l2Support.toBadge(context),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
if (isDropdown)
|
|
|
|
|
Icon(
|
|
|
|
|
Icons.arrow_drop_down,
|
|
|
|
|
color: Theme.of(context).colorScheme.onSurfaceVariant,
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|