mirror of
https://git.selfprivacy.org/kherel/selfprivacy.org.app.git
synced 2025-01-25 18:26:36 +00:00
feat(timezone): Implement search bar for 'Select Timezone' page
This commit is contained in:
parent
e1419ce38f
commit
e36a94ded5
|
@ -101,6 +101,7 @@
|
||||||
"reboot_after_upgrade_hint": "Reboot without prompt after applying changes on server",
|
"reboot_after_upgrade_hint": "Reboot without prompt after applying changes on server",
|
||||||
"server_timezone": "Server timezone",
|
"server_timezone": "Server timezone",
|
||||||
"select_timezone": "Select timezone",
|
"select_timezone": "Select timezone",
|
||||||
|
"timezone_search_bar": "Timezone name or time shift value",
|
||||||
"server_id": "Server ID",
|
"server_id": "Server ID",
|
||||||
"status": "Status",
|
"status": "Status",
|
||||||
"cpu": "CPU",
|
"cpu": "CPU",
|
||||||
|
|
|
@ -101,6 +101,7 @@
|
||||||
"reboot_after_upgrade_hint": "Автоматически перезагружать сервер после применения обновлений",
|
"reboot_after_upgrade_hint": "Автоматически перезагружать сервер после применения обновлений",
|
||||||
"server_timezone": "Часовой пояс сервера",
|
"server_timezone": "Часовой пояс сервера",
|
||||||
"select_timezone": "Выберите часовой пояс",
|
"select_timezone": "Выберите часовой пояс",
|
||||||
|
"timezone_search_bar": "Имя часового пояса или значение временного сдвига",
|
||||||
"server_id": "ID сервера",
|
"server_id": "ID сервера",
|
||||||
"status": "Статус",
|
"status": "Статус",
|
||||||
"cpu": "Процессор",
|
"cpu": "Процессор",
|
||||||
|
|
|
@ -12,6 +12,7 @@ import 'package:selfprivacy/logic/models/auto_upgrade_settings.dart';
|
||||||
import 'package:selfprivacy/logic/models/job.dart';
|
import 'package:selfprivacy/logic/models/job.dart';
|
||||||
import 'package:selfprivacy/ui/components/brand_button/segmented_buttons.dart';
|
import 'package:selfprivacy/ui/components/brand_button/segmented_buttons.dart';
|
||||||
import 'package:selfprivacy/ui/components/brand_cards/filled_card.dart';
|
import 'package:selfprivacy/ui/components/brand_cards/filled_card.dart';
|
||||||
|
import 'package:selfprivacy/ui/components/brand_header/brand_header.dart';
|
||||||
import 'package:selfprivacy/ui/components/brand_hero_screen/brand_hero_screen.dart';
|
import 'package:selfprivacy/ui/components/brand_hero_screen/brand_hero_screen.dart';
|
||||||
import 'package:selfprivacy/ui/components/brand_icons/brand_icons.dart';
|
import 'package:selfprivacy/ui/components/brand_icons/brand_icons.dart';
|
||||||
import 'package:selfprivacy/ui/components/brand_loader/brand_loader.dart';
|
import 'package:selfprivacy/ui/components/brand_loader/brand_loader.dart';
|
||||||
|
|
|
@ -14,11 +14,20 @@ class SelectTimezone extends StatefulWidget {
|
||||||
}
|
}
|
||||||
|
|
||||||
class _SelectTimezoneState extends State<SelectTimezone> {
|
class _SelectTimezoneState extends State<SelectTimezone> {
|
||||||
final ScrollController controller = ScrollController();
|
final ScrollController scrollController = ScrollController();
|
||||||
|
final TextEditingController searchController = TextEditingController();
|
||||||
|
|
||||||
|
String? timezoneFilterValue;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
WidgetsBinding.instance.addPostFrameCallback(_afterLayout);
|
WidgetsBinding.instance.addPostFrameCallback(_afterLayout);
|
||||||
|
searchController.addListener(() {
|
||||||
|
setState(() {
|
||||||
|
timezoneFilterValue =
|
||||||
|
searchController.text.isNotEmpty ? searchController.text : null;
|
||||||
|
});
|
||||||
|
});
|
||||||
super.initState();
|
super.initState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,7 +40,7 @@ class _SelectTimezoneState extends State<SelectTimezone> {
|
||||||
print(t);
|
print(t);
|
||||||
|
|
||||||
if (index >= 0) {
|
if (index >= 0) {
|
||||||
controller.animateTo(
|
scrollController.animateTo(
|
||||||
60.0 * index,
|
60.0 * index,
|
||||||
duration: const Duration(milliseconds: 300),
|
duration: const Duration(milliseconds: 300),
|
||||||
curve: Curves.easeIn,
|
curve: Curves.easeIn,
|
||||||
|
@ -41,93 +50,122 @@ class _SelectTimezoneState extends State<SelectTimezone> {
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void dispose() {
|
void dispose() {
|
||||||
controller.dispose();
|
scrollController.dispose();
|
||||||
|
searchController.dispose();
|
||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(final BuildContext context) => Scaffold(
|
Widget build(final BuildContext context) => Scaffold(
|
||||||
appBar: AppBar(
|
appBar: PreferredSize(
|
||||||
title: Padding(
|
preferredSize: const Size.fromHeight(156),
|
||||||
padding: const EdgeInsets.only(top: 4.0),
|
child: Column(
|
||||||
child: Text('server.select_timezone'.tr()),
|
children: [
|
||||||
),
|
BrandHeader(
|
||||||
leading: IconButton(
|
title: 'server.select_timezone'.tr(),
|
||||||
icon: const Icon(Icons.arrow_back),
|
),
|
||||||
onPressed: () => Navigator.of(context).pop(),
|
Row(
|
||||||
|
children: [
|
||||||
|
const SizedBox(width: 16),
|
||||||
|
SizedBox(
|
||||||
|
height: 52,
|
||||||
|
child: TextField(
|
||||||
|
readOnly: false,
|
||||||
|
textAlign: TextAlign.start,
|
||||||
|
textInputAction: TextInputAction.next,
|
||||||
|
enabled: true,
|
||||||
|
controller: searchController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
errorText: null,
|
||||||
|
labelText: 'server.timezone_search_bar'.tr(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 16),
|
||||||
|
const Icon(Icons.search_outlined),
|
||||||
|
const SizedBox(width: 16),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
body: SafeArea(
|
body: SafeArea(
|
||||||
child: ListView(
|
child: ListView(
|
||||||
controller: controller,
|
controller: scrollController,
|
||||||
children: locations
|
children: locations
|
||||||
|
.where(
|
||||||
|
(final Location location) => timezoneFilterValue == null
|
||||||
|
? true
|
||||||
|
: location.name
|
||||||
|
.toLowerCase()
|
||||||
|
.contains(timezoneFilterValue!) ||
|
||||||
|
Duration(
|
||||||
|
milliseconds: location.currentTimeZone.offset,
|
||||||
|
)
|
||||||
|
.toDayHourMinuteFormat()
|
||||||
|
.contains(timezoneFilterValue!),
|
||||||
|
)
|
||||||
|
.toList()
|
||||||
.asMap()
|
.asMap()
|
||||||
.map((final key, final value) {
|
.map(
|
||||||
final duration =
|
(final key, final value) => locationToListTile(key, value),
|
||||||
Duration(milliseconds: value.currentTimeZone.offset);
|
)
|
||||||
final area = value.currentTimeZone.abbreviation
|
|
||||||
.replaceAll(RegExp(r'[\d+()-]'), '');
|
|
||||||
|
|
||||||
String timezoneName = value.name;
|
|
||||||
if (context.locale.toString() == 'ru') {
|
|
||||||
timezoneName = russian[value.name] ??
|
|
||||||
() {
|
|
||||||
final arr = value.name.split('/')..removeAt(0);
|
|
||||||
return arr.join('/');
|
|
||||||
}();
|
|
||||||
}
|
|
||||||
|
|
||||||
return MapEntry(
|
|
||||||
key,
|
|
||||||
Container(
|
|
||||||
height: 75,
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 20),
|
|
||||||
decoration: const BoxDecoration(
|
|
||||||
border: Border(
|
|
||||||
bottom: BorderSide(
|
|
||||||
color: BrandColors.dividerColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: InkWell(
|
|
||||||
onTap: () {
|
|
||||||
context
|
|
||||||
.read<ServerDetailsCubit>()
|
|
||||||
.repository
|
|
||||||
.setTimezone(
|
|
||||||
timezoneName,
|
|
||||||
);
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
child: Container(
|
|
||||||
padding: const EdgeInsets.symmetric(vertical: 16),
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
BrandText.body1(
|
|
||||||
timezoneName,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontSize: 16,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
BrandText.small(
|
|
||||||
'GMT ${duration.toDayHourMinuteFormat()} ${area.isNotEmpty ? '($area)' : ''}',
|
|
||||||
style: const TextStyle(
|
|
||||||
fontSize: 13,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
})
|
|
||||||
.values
|
.values
|
||||||
.toList(),
|
.toList(),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
MapEntry<int, Container> locationToListTile(
|
||||||
|
final int key, final Location location) {
|
||||||
|
final duration = Duration(milliseconds: location.currentTimeZone.offset);
|
||||||
|
final area = location.currentTimeZone.abbreviation
|
||||||
|
.replaceAll(RegExp(r'[\d+()-]'), '');
|
||||||
|
|
||||||
|
return MapEntry(
|
||||||
|
key,
|
||||||
|
Container(
|
||||||
|
height: 75,
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 20),
|
||||||
|
decoration: const BoxDecoration(
|
||||||
|
border: Border(
|
||||||
|
bottom: BorderSide(
|
||||||
|
color: BrandColors.dividerColor,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
child: InkWell(
|
||||||
|
onTap: () {
|
||||||
|
context.read<ServerDetailsCubit>().repository.setTimezone(
|
||||||
|
location.name,
|
||||||
|
);
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
},
|
||||||
|
child: Container(
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 16),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
BrandText.body1(
|
||||||
|
location.name,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontSize: 16,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
BrandText.small(
|
||||||
|
'GMT ${duration.toDayHourMinuteFormat()} ${area.isNotEmpty ? '($area)' : ''}',
|
||||||
|
style: const TextStyle(
|
||||||
|
fontSize: 13,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,6 +32,13 @@ class _ExtendingVolumePageState extends State<ExtendingVolumePage> {
|
||||||
super.initState();
|
super.initState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_sizeController.dispose();
|
||||||
|
_priceController.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
bool _isError = false;
|
bool _isError = false;
|
||||||
|
|
||||||
late double _currentSliderGbValue;
|
late double _currentSliderGbValue;
|
||||||
|
|
Loading…
Reference in a new issue