18 Commits
lab5 ... lab7

Author SHA1 Message Date
235cf4b331 Поправил локализацию для SnackBar 2024-12-21 14:46:49 +04:00
8132957bf2 refactor: убрал закоментированный код 2024-12-21 14:35:04 +04:00
c087b6a2a2 Сменил иконку и пофиксил баг с прокруткой до конца 2024-12-19 19:27:53 +04:00
4d3d485e4d Пофиксил лайк 2024-12-19 18:59:49 +04:00
569a2aff06 Надо починить лайки 2024-12-19 18:44:53 +04:00
60554238e3 начал делать svg 2024-12-19 17:18:35 +04:00
b276cecb1a Добавил классы для хранения лайков 2024-12-19 14:59:38 +04:00
20eae72ab0 Добавил классы для локализации 2024-12-19 14:49:26 +04:00
e7c9e97f21 добавил команды в Makefile 2024-12-19 14:28:34 +04:00
a625743279 добавил зависимости в pubspec.yaml 2024-12-19 14:26:56 +04:00
77a8eb3147 Готова 6 лаба 2024-12-19 13:20:27 +04:00
41926b4190 Поправил api_interface.dart 2024-12-17 15:18:01 +04:00
6d3d79022c Правки main.dart 2024-12-13 04:00:50 +04:00
1d5332db3c Добавил классы bloc для home page 2024-12-13 03:41:33 +04:00
906a889c04 Добавил фильтрацию в репозиторий 2024-12-13 03:36:58 +04:00
4d5291471b Добавил BLoC зависимости в проект 2024-12-13 03:25:26 +04:00
4eb414293e Создал класс дебаунса 2024-12-13 03:20:11 +04:00
331032cbec Добавил png в разрешенные форматы маппера 2024-12-13 03:19:59 +04:00
46 changed files with 1421 additions and 182 deletions

View File

@@ -1,26 +1,24 @@
# Устанавливаем переменную для команды flutter gen:
FLUTTER = flutter flutter pub run build_runner build --delete-conflicting-outputs
# Команда для генерации кода с build_runner hello:
gen: install-dependencies echo "Hi!"; \
$(FLUTTER) pub run build_runner build --delete-conflicting-outputs echo "I'm makefile"; \
echo "^_^"
# Устанавливаем зависимости (если их нет) icon:
install-dependencies: flutter pub run flutter_launcher_icons:main
$(FLUTTER) pub get
# Очистка проекта (по желанию) init_res:
clean: dart pub global activate flutter_asset_generator
$(FLUTTER) clean
# Для генерации кода с очисткой format:
clean-generate: clean generate dart format . --line-length 100
# Отображение доступных команд res:
help: fgen --output lib/components/resources.g.dart --no-watch --no-preview; \
@echo "Доступные команды:" make format
@echo " generate - Запуск генерации кода с build_runner"
@echo " install-dependencies - Установить зависимости" loc:
@echo " clean - Очистить проект" flutter gen-l10n; \
@echo " clean-generate - Очистка и генерация кода" make format
@echo " help - Показать эту справку"

Binary file not shown.

Before

Width:  |  Height:  |  Size: 544 B

After

Width:  |  Height:  |  Size: 12 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 442 B

After

Width:  |  Height:  |  Size: 5.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 721 B

After

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.0 KiB

After

Width:  |  Height:  |  Size: 44 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

After

Width:  |  Height:  |  Size: 76 KiB

BIN
assets/launcher.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 MiB

10
assets/svg/ru.svg Normal file
View File

@@ -0,0 +1,10 @@
<svg width="210" height="210" viewBox="0 0 210 210" fill="none" xmlns="http://www.w3.org/2000/svg">
<mask id="mask0_2_16" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="0" y="0" width="210" height="210">
<circle cx="105" cy="105" r="105" fill="#D9D9D9"/>
</mask>
<g mask="url(#mask0_2_16)">
<rect width="210" height="70" fill="white"/>
<rect y="70" width="210" height="70" fill="#92BCEC"/>
<rect y="140" width="210" height="70" fill="#FF6A59"/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 464 B

16
assets/svg/uk.svg Normal file
View File

@@ -0,0 +1,16 @@
<svg width="210" height="210" viewBox="0 0 210 210" fill="none" xmlns="http://www.w3.org/2000/svg">
<mask id="mask0_2_33" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="0" y="0" width="210" height="210">
<circle cx="105" cy="105" r="105" fill="#D9D9D9"/>
</mask>
<g mask="url(#mask0_2_33)">
<rect x="-45.178" y="-2.80414" width="291.632" height="210" fill="#92BCEC"/>
<path d="M-53.9021 235.549L302.849 -25.8606" stroke="white" stroke-width="31.1573"/>
<path d="M252.507 234.39L-93.4718 -29.2878" stroke="white" stroke-width="31.1573"/>
<line x1="103.131" y1="-35.8309" x2="103.131" y2="240.846" stroke="white" stroke-width="43.6202"/>
<path d="M-97.8338 118.086L246.454 118.086" stroke="white" stroke-width="43.6202"/>
<line x1="309.671" y1="-33.314" x2="-59.2315" y2="238.377" stroke="#FF6A59" stroke-width="9.34718"/>
<path d="M-65.6983 -7.13381L301.914 270.757" stroke="#FF6A59" stroke-width="9.34718"/>
<line x1="103.131" y1="-45.8012" x2="103.131" y2="260.786" stroke="#FF6A59" stroke-width="24.9258"/>
<path d="M-62.6261 118.086L308.145 118.086" stroke="#FF6A59" stroke-width="24.9258"/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 1.1 KiB

3
devtools_options.yaml Normal file
View File

@@ -0,0 +1,3 @@
description: This file stores settings for Dart & Flutter DevTools.
documentation: https://docs.flutter.dev/tools/devtools/extensions#configure-extension-enablement-states
extensions:

6
l10n.yaml Normal file
View File

@@ -0,0 +1,6 @@
arb-dir: l10n
template-arb-file: app_ru.arb
output-localization-file: app_locale.dart
output-dir: lib/components/locale/l10n
output-class: AppLocale
synthetic-package: false

9
l10n/app_en.arb Normal file
View File

@@ -0,0 +1,9 @@
{
"@@locale": "en",
"search": "Search",
"liked": "You liked:",
"disliked": "You disliked:",
"arbEnding": "Чтобы не забыть про отсутствие запятой :)"
}

9
l10n/app_ru.arb Normal file
View File

@@ -0,0 +1,9 @@
{
"@@locale": "ru",
"search": "Поиск",
"liked": "Вы поставили лайк:",
"disliked": "Вы убрали лайк:",
"arbEnding": "Чтобы не забыть про отсутствие запятой :)"
}

View File

@@ -0,0 +1,18 @@
import 'dart:async';
import 'dart:ui';
class Debounce {
factory Debounce() => _instance;
Debounce._();
static final Debounce _instance = Debounce._();
static Timer? _timer;
static void run(VoidCallback action,
{Duration delay = const Duration(milliseconds: 500)}) {
_timer?.cancel();
_timer = Timer(delay, action);
}
}

View File

@@ -0,0 +1,6 @@
import 'package:flutter/widgets.dart';
import 'package:pmu/components/locale/l10n/app_locale.dart';
extension LocalContextX on BuildContext {
AppLocale get locale => AppLocale.of(this)!;
}

View File

@@ -0,0 +1,153 @@
import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:flutter/widgets.dart';
import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:intl/intl.dart' as intl;
import 'app_locale_en.dart';
import 'app_locale_ru.dart';
// ignore_for_file: type=lint
/// Callers can lookup localized strings with an instance of AppLocale
/// returned by `AppLocale.of(context)`.
///
/// Applications need to include `AppLocale.delegate()` in their app's
/// `localizationDelegates` list, and the locales they support in the app's
/// `supportedLocales` list. For example:
///
/// ```dart
/// import 'l10n/app_locale.dart';
///
/// return MaterialApp(
/// localizationsDelegates: AppLocale.localizationsDelegates,
/// supportedLocales: AppLocale.supportedLocales,
/// home: MyApplicationHome(),
/// );
/// ```
///
/// ## Update pubspec.yaml
///
/// Please make sure to update your pubspec.yaml to include the following
/// packages:
///
/// ```yaml
/// dependencies:
/// # Internationalization support.
/// flutter_localizations:
/// sdk: flutter
/// intl: any # Use the pinned version from flutter_localizations
///
/// # Rest of dependencies
/// ```
///
/// ## iOS Applications
///
/// iOS applications define key application metadata, including supported
/// locales, in an Info.plist file that is built into the application bundle.
/// To configure the locales supported by your app, youll need to edit this
/// file.
///
/// First, open your projects ios/Runner.xcworkspace Xcode workspace file.
/// Then, in the Project Navigator, open the Info.plist file under the Runner
/// projects Runner folder.
///
/// Next, select the Information Property List item, select Add Item from the
/// Editor menu, then select Localizations from the pop-up menu.
///
/// Select and expand the newly-created Localizations item then, for each
/// locale your application supports, add a new item and select the locale
/// you wish to add from the pop-up menu in the Value field. This list should
/// be consistent with the languages listed in the AppLocale.supportedLocales
/// property.
abstract class AppLocale {
AppLocale(String locale) : localeName = intl.Intl.canonicalizedLocale(locale.toString());
final String localeName;
static AppLocale? of(BuildContext context) {
return Localizations.of<AppLocale>(context, AppLocale);
}
static const LocalizationsDelegate<AppLocale> delegate = _AppLocaleDelegate();
/// A list of this localizations delegate along with the default localizations
/// delegates.
///
/// Returns a list of localizations delegates containing this delegate along with
/// GlobalMaterialLocalizations.delegate, GlobalCupertinoLocalizations.delegate,
/// and GlobalWidgetsLocalizations.delegate.
///
/// Additional delegates can be added by appending to this list in
/// MaterialApp. This list does not have to be used at all if a custom list
/// of delegates is preferred or required.
static const List<LocalizationsDelegate<dynamic>> localizationsDelegates = <LocalizationsDelegate<dynamic>>[
delegate,
GlobalMaterialLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
];
/// A list of this localizations delegate's supported locales.
static const List<Locale> supportedLocales = <Locale>[
Locale('en'),
Locale('ru')
];
/// No description provided for @search.
///
/// In ru, this message translates to:
/// **'Поиск'**
String get search;
/// No description provided for @liked.
///
/// In ru, this message translates to:
/// **'Вы поставили лайк:'**
String get liked;
/// No description provided for @disliked.
///
/// In ru, this message translates to:
/// **'Вы убрали лайк:'**
String get disliked;
/// No description provided for @arbEnding.
///
/// In ru, this message translates to:
/// **'Чтобы не забыть про отсутствие запятой :)'**
String get arbEnding;
}
class _AppLocaleDelegate extends LocalizationsDelegate<AppLocale> {
const _AppLocaleDelegate();
@override
Future<AppLocale> load(Locale locale) {
return SynchronousFuture<AppLocale>(lookupAppLocale(locale));
}
@override
bool isSupported(Locale locale) => <String>['en', 'ru'].contains(locale.languageCode);
@override
bool shouldReload(_AppLocaleDelegate old) => false;
}
AppLocale lookupAppLocale(Locale locale) {
// Lookup logic when only language code is specified.
switch (locale.languageCode) {
case 'en': return AppLocaleEn();
case 'ru': return AppLocaleRu();
}
throw FlutterError(
'AppLocale.delegate failed to load unsupported locale "$locale". This is likely '
'an issue with the localizations generation tool. Please file an issue '
'on GitHub with a reproducible sample app and the gen-l10n configuration '
'that was used.'
);
}

View File

@@ -0,0 +1,20 @@
import 'app_locale.dart';
// ignore_for_file: type=lint
/// The translations for English (`en`).
class AppLocaleEn extends AppLocale {
AppLocaleEn([String locale = 'en']) : super(locale);
@override
String get search => 'Search';
@override
String get liked => 'You liked:';
@override
String get disliked => 'You disliked:';
@override
String get arbEnding => 'Чтобы не забыть про отсутствие запятой :)';
}

View File

@@ -0,0 +1,20 @@
import 'app_locale.dart';
// ignore_for_file: type=lint
/// The translations for Russian (`ru`).
class AppLocaleRu extends AppLocale {
AppLocaleRu([String locale = 'ru']) : super(locale);
@override
String get search => 'Поиск';
@override
String get liked => 'Вы поставили лайк:';
@override
String get disliked => 'Вы убрали лайк:';
@override
String get arbEnding => 'Чтобы не забыть про отсутствие запятой :)';
}

View File

@@ -0,0 +1,12 @@
/// Generate by [asset_generator](https://github.com/fluttercandies/flutter_asset_generator) library.
/// PLEASE DO NOT EDIT MANUALLY.
// ignore_for_file: constant_identifier_names
class R {
const R._();
/// ![preview](file://D:\code\pmu_nikita\PMU-PIbd-31-Potapov-N-S\assets\svg\ru.svg)
static const String ASSETS_SVG_RU_SVG = 'assets/svg/ru.svg';
/// ![preview](file://D:\code\pmu_nikita\PMU-PIbd-31-Potapov-N-S\assets\svg\uk.svg)
static const String ASSETS_SVG_UK_SVG = 'assets/svg/uk.svg';
}

View File

@@ -0,0 +1,12 @@
/// Generate by [asset_generator](https://github.com/fluttercandies/flutter_asset_generator) library.
/// PLEASE DO NOT EDIT MANUALLY.
// ignore_for_file: constant_identifier_names
class R {
const R._();
/// ![preview](file://D:\code\pmu_labs\pmu_labs_3_course\assets\svg\ru.svg)
static const String ASSETS_SVG_RU_SVG = 'assets/svg/ru.svg';
/// ![preview](file://D:\code\pmu_labs\pmu_labs_3_course\assets\svg\uk.svg)
static const String ASSETS_SVG_UK_SVG = 'assets/svg/uk.svg';
}

View File

@@ -1,9 +1,9 @@
import 'package:pmu/data/dto/user_dto.dart'; import 'package:pmu/data/dto/user_dto.dart';
import 'package:pmu/domain/card.dart'; import 'package:pmu/domain/models/card.dart';
extension UserDtoToModel on UserDto { extension UserDtoToModel on UserDto {
CardPostData toDomain() { CardData toDomain() {
const allowedExtensions = ['jpg', 'jpeg', 'gif']; const allowedExtensions = ['jpg', 'jpeg', 'gif', 'png'];
bool isValidImageUrl(String? url) { bool isValidImageUrl(String? url) {
if (url == null) return false; if (url == null) return false;
@@ -11,7 +11,7 @@ extension UserDtoToModel on UserDto {
return allowedExtensions.contains(extension); return allowedExtensions.contains(extension);
} }
return CardPostData( return CardData(
name: name ?? "", name: name ?? "",
surname: surname ?? "", surname: surname ?? "",
description: description ?? "", description: description ?? "",
@@ -21,8 +21,8 @@ extension UserDtoToModel on UserDto {
? "https://avatar.iran.liara.run/public/boy" ? "https://avatar.iran.liara.run/public/boy"
: "https://avatar.iran.liara.run/public/girl") : "https://avatar.iran.liara.run/public/girl")
.toString(), .toString(),
isLiked: false,
age: age, age: age,
distance: distance); distance: distance,
id: id?.toString());
} }
} }

View File

@@ -1,5 +1,7 @@
import 'package:pmu/domain/card.dart'; import 'package:pmu/domain/models/home.dart';
typedef OnErrorCallback = void Function(String? error);
abstract class ApiInterface { abstract class ApiInterface {
Future<List<CardPostData>?> loadData(); Future<HomeData?> loadData({OnErrorCallback? onError});
} }

View File

@@ -0,0 +1,54 @@
import 'package:pmu/data/dto/page_dto.dart';
import 'package:pmu/data/mappers/user_mapper.dart';
import 'package:pmu/data/repositories/api_interface.dart';
import 'package:pmu/domain/models/card.dart';
import 'package:dio/dio.dart';
import 'package:pmu/domain/models/home.dart';
import 'package:pretty_dio_logger/pretty_dio_logger.dart';
class ApiRepository extends ApiInterface {
static final Dio _dio = Dio()
..interceptors.add(PrettyDioLogger(requestHeader: true, requestBody: true));
static const String _baseUrl = 'http://lovesearch-api.nspotapov.ru';
@override
Future<HomeData?> loadData({
OnErrorCallback? onError,
String? q,
int pageNumber = 0,
int itemsByPage = 5,
}) async {
try {
const String url = '$_baseUrl/api/users/';
final List<CardData> data = [];
Map<String, dynamic> queryParameters = {
'pageNumber': pageNumber,
'itemsByPage': itemsByPage
};
if (q != null) {
queryParameters['q'] = q;
}
final Response<dynamic> response = await _dio
.get<Map<dynamic, dynamic>>(url, queryParameters: queryParameters);
final PageDto pageDto =
PageDto.fromJson(response.data as Map<String, dynamic>);
int itemsCount = pageDto.itemsCount ?? 0;
for (int i = 0; i < itemsCount; i++) {
final CardData cardPost = pageDto.items![i].toDomain();
data.add(cardPost);
}
HomeData homeData = HomeData(
data: data,
pageNumber: pageDto.pageNumber,
nextPageNumber: pageDto.nextPageNumber,
lastPageNumber: pageDto.lastPageNumber);
return homeData;
} on DioException catch (e) {
onError?.call(e.response?.statusMessage);
onError?.call(e.error?.toString());
return null;
}
}
}

View File

@@ -1,30 +0,0 @@
import 'package:pmu/data/dto/page_dto.dart';
import 'package:pmu/data/dto/user_dto.dart';
import 'package:pmu/data/mappers/user_mapper.dart';
import 'package:pmu/data/repositories/api_interface.dart';
import 'package:pmu/domain/card.dart';
import 'package:dio/dio.dart';
import 'package:pretty_dio_logger/pretty_dio_logger.dart';
class ApiUserRepository extends ApiInterface {
static final Dio _dio = Dio()
..interceptors.add(PrettyDioLogger(requestHeader: true, requestBody: true));
static const String _baseUrl =
'http://lovesearch-api.nspotapov.ru/api/users/';
@override
Future<List<CardPostData>?> loadData() async {
const String url = _baseUrl;
final List<CardPostData> data = [];
final Response<dynamic> response =
await _dio.get<Map<dynamic, dynamic>>(url);
final PageDto pageDto =
PageDto.fromJson(response.data as Map<String, dynamic>);
int itemsCount = pageDto.itemsCount ?? 0;
for (int i = 0; i < itemsCount; i++) {
final CardPostData cardPost = pageDto.items![i].toDomain();
data.add(cardPost);
}
return data;
}
}

View File

@@ -1,26 +1,27 @@
import 'package:pmu/data/repositories/api_interface.dart'; import 'package:pmu/data/repositories/api_interface.dart';
import 'package:pmu/domain/card.dart'; import 'package:pmu/domain/models/card.dart';
import 'package:pmu/domain/models/home.dart';
class MockUserRepository extends ApiInterface { class MockRepository extends ApiInterface {
@override @override
Future<List<CardPostData>?> loadData() async { Future<HomeData?> loadData({OnErrorCallback? onError}) async {
return [ return HomeData(data: [
const CardPostData( const CardData(
name: "Марьяна", name: "Марьяна",
surname: "Ро", surname: "Ро",
description: "Люблю вечерами погамать в майн", description: "Люблю вечерами погамать в майн",
imageUrl: "https://placehold.co/600x400/png", imageUrl: "https://placehold.co/600x400/png",
age: 21, age: 21,
distance: 24.5, distance: 24.5,
isLiked: false), ),
const CardPostData( const CardData(
name: "Константин", name: "Константин",
surname: "Злобин", surname: "Злобин",
description: "Веду канал в тг про криптожизнь", description: "Веду канал в тг про криптожизнь",
imageUrl: "https://placehold.co/600x400/png", imageUrl: "https://placehold.co/600x400/png",
age: 24, age: 24,
distance: 478.3, distance: 478.3,
isLiked: false) )
]; ], pageNumber: 0, lastPageNumber: 0);
} }
} }

View File

@@ -1,18 +1,21 @@
class CardPostData { class CardData {
final String? name; final String? name;
final String? surname; final String? surname;
final String? description; final String? description;
final String? imageUrl; final String? imageUrl;
final int? age; final int? age;
final double? distance; final double? distance;
final bool? isLiked;
const CardPostData( final String? id;
const CardData(
{this.name, {this.name,
this.surname, this.surname,
this.description, this.description,
this.imageUrl, this.imageUrl,
this.isLiked,
this.age, this.age,
this.distance}); this.distance,
this.id});
} }

View File

@@ -0,0 +1,10 @@
import 'card.dart';
class HomeData {
final List<CardData>? data;
final int? pageNumber;
final int? nextPageNumber;
final int? lastPageNumber;
HomeData({this.data, this.pageNumber, this.nextPageNumber, this.lastPageNumber});
}

View File

@@ -1,5 +1,15 @@
import 'dart:io';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:pmu/data/repositories/api_repository.dart';
import 'package:pmu/presentation/home_page/bloc/bloc.dart';
import 'package:pmu/presentation/home_page/home_page.dart'; import 'package:pmu/presentation/home_page/home_page.dart';
import 'package:pmu/presentation/like/bloc.dart';
import 'package:pmu/presentation/locale/bloc.dart';
import 'package:pmu/presentation/locale/state.dart';
import 'components/locale/l10n/app_locale.dart';
void main() { void main() {
runApp(const MyApp()); runApp(const MyApp());
@@ -10,9 +20,33 @@ class MyApp extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return const MaterialApp( return BlocProvider<LocaleBloc>(
title: 'Meets', lazy: false,
home: MyHomePage(title: 'Знакомства'), create: (context) => LocaleBloc(Locale(Platform.localeName)),
child: BlocBuilder<LocaleBloc, LocaleState>(
builder: (context, state) {
return MaterialApp(
title: 'LoveSearch',
locale: state.currentLocale,
localizationsDelegates: AppLocale.localizationsDelegates,
supportedLocales: AppLocale.supportedLocales,
debugShowCheckedModeBanner: false,
home: RepositoryProvider<ApiRepository>(
lazy: true,
create: (_) => ApiRepository(),
child: BlocProvider<LikeBloc>(
lazy: true,
create: (context) => LikeBloc(),
child: BlocProvider<HomeBloc>(
lazy: false,
create: (context) =>
HomeBloc(context.read<ApiRepository>()),
child: const HomePage()),
),
),
);
},
),
); );
} }
} }

View File

@@ -0,0 +1,36 @@
import 'package:flutter/widgets.dart';
import 'package:flutter_svg/flutter_svg.dart';
import '../../components/resoures.g.dart';
abstract class SvgObjects {
static void init() {
final pics = <String>[
R.ASSETS_SVG_RU_SVG,
R.ASSETS_SVG_UK_SVG,
];
for (final String p in pics) {
final loader = SvgAssetLoader(p);
svg.cache
.putIfAbsent(loader.cacheKey(null), () => loader.loadBytes(null));
}
}
}
class SvgRu extends StatelessWidget {
const SvgRu({super.key});
@override
Widget build(BuildContext context) {
return SvgPicture.asset(R.ASSETS_SVG_RU_SVG);
}
}
class SvgUk extends StatelessWidget {
const SvgUk({super.key});
@override
Widget build(BuildContext context) {
return SvgPicture.asset(R.ASSETS_SVG_UK_SVG);
}
}

View File

@@ -1,9 +1,9 @@
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:pmu/domain/card.dart'; import 'package:pmu/domain/models/card.dart';
class DetailPage extends StatelessWidget { class DetailPage extends StatelessWidget {
final CardPostData data; final CardData data;
const DetailPage(this.data, {super.key}); const DetailPage(this.data, {super.key});

View File

@@ -0,0 +1,51 @@
import 'dart:async';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:pmu/data/repositories/api_repository.dart';
import 'package:pmu/presentation/home_page/bloc/events.dart';
import 'package:pmu/presentation/home_page/bloc/state.dart';
class HomeBloc extends Bloc<HomeEvent, HomeState> {
final ApiRepository repository;
HomeBloc(this.repository) : super(const HomeState()) {
on<HomeLoadDataEvent>(_onLoadData);
}
Future<void> _onLoadData(
HomeLoadDataEvent event, Emitter<HomeState> emit) async {
if (event.pageNumber == event.lastPageNumber && event.pageNumber != null) {
emit(state.copyWith(
isLoading: false,
isPaginationLoading: false,
));
return;
}
if (event.nextPageNumber == null) {
emit(state.copyWith(isLoading: true));
} else {
emit(state.copyWith(isPaginationLoading: true));
}
String? error;
final data = await repository.loadData(
q: event.search,
pageNumber: event.nextPageNumber ?? 0,
onError: (e) => error = e,
);
if (event.nextPageNumber != null) {
data?.data?.insertAll(0, state.data?.data ?? []);
}
emit(state.copyWith(
isLoading: false,
isPaginationLoading: false,
data: data,
error: error,
));
}
}

View File

@@ -0,0 +1,13 @@
abstract class HomeEvent {
const HomeEvent();
}
class HomeLoadDataEvent extends HomeEvent {
final String? search;
final int? pageNumber;
final int? nextPageNumber;
final int? lastPageNumber;
const HomeLoadDataEvent(
{this.search, this.pageNumber, this.nextPageNumber, this.lastPageNumber});
}

View File

@@ -0,0 +1,28 @@
import 'package:copy_with_extension/copy_with_extension.dart';
import 'package:equatable/equatable.dart';
import 'package:pmu/domain/models/home.dart';
part 'state.g.dart';
@CopyWith()
class HomeState extends Equatable {
final HomeData? data;
final bool isLoading;
final bool isPaginationLoading;
final String? error;
const HomeState({
this.data,
this.isLoading = false,
this.isPaginationLoading = false,
this.error,
});
@override
List<Object?> get props => [
data,
isLoading,
isPaginationLoading,
error,
];
}

View File

@@ -0,0 +1,90 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'state.dart';
// **************************************************************************
// CopyWithGenerator
// **************************************************************************
abstract class _$HomeStateCWProxy {
HomeState data(HomeData? data);
HomeState isLoading(bool isLoading);
HomeState isPaginationLoading(bool isPaginationLoading);
HomeState error(String? error);
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `HomeState(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// HomeState(...).copyWith(id: 12, name: "My name")
/// ````
HomeState call({
HomeData? data,
bool isLoading,
bool isPaginationLoading,
String? error,
});
}
/// Proxy class for `copyWith` functionality. This is a callable class and can be used as follows: `instanceOfHomeState.copyWith(...)`. Additionally contains functions for specific fields e.g. `instanceOfHomeState.copyWith.fieldName(...)`
class _$HomeStateCWProxyImpl implements _$HomeStateCWProxy {
const _$HomeStateCWProxyImpl(this._value);
final HomeState _value;
@override
HomeState data(HomeData? data) => this(data: data);
@override
HomeState isLoading(bool isLoading) => this(isLoading: isLoading);
@override
HomeState isPaginationLoading(bool isPaginationLoading) =>
this(isPaginationLoading: isPaginationLoading);
@override
HomeState error(String? error) => this(error: error);
@override
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `HomeState(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// HomeState(...).copyWith(id: 12, name: "My name")
/// ````
HomeState call({
Object? data = const $CopyWithPlaceholder(),
Object? isLoading = const $CopyWithPlaceholder(),
Object? isPaginationLoading = const $CopyWithPlaceholder(),
Object? error = const $CopyWithPlaceholder(),
}) {
return HomeState(
data: data == const $CopyWithPlaceholder()
? _value.data
// ignore: cast_nullable_to_non_nullable
: data as HomeData?,
isLoading: isLoading == const $CopyWithPlaceholder()
? _value.isLoading
// ignore: cast_nullable_to_non_nullable
: isLoading as bool,
isPaginationLoading: isPaginationLoading == const $CopyWithPlaceholder()
? _value.isPaginationLoading
// ignore: cast_nullable_to_non_nullable
: isPaginationLoading as bool,
error: error == const $CopyWithPlaceholder()
? _value.error
// ignore: cast_nullable_to_non_nullable
: error as String?,
);
}
}
extension $HomeStateCopyWith on HomeState {
/// Returns a callable class that can be used as follows: `instanceOfHomeState.copyWith(...)` or like so:`instanceOfHomeState.copyWith.fieldName(...)`.
// ignore: library_private_types_in_public_api
_$HomeStateCWProxy get copyWith => _$HomeStateCWProxyImpl(this);
}

View File

@@ -1,56 +1,48 @@
import 'dart:async'; part of 'home_page.dart';
import 'package:flutter/material.dart'; typedef OnLikeCallback = void Function(String? id, String title, bool isLiked)?;
import 'package:pmu/domain/card.dart';
typedef OnLikeCallback = void Function(String title, bool isLiked)?;
const double NORMAL_ICON_SCALE = 2.0; const double NORMAL_ICON_SCALE = 2.0;
const double SCALED_ICON_SCALE = 2.5; const double SCALED_ICON_SCALE = 2.5;
class CardPost extends StatefulWidget { class _Card extends StatelessWidget {
final String? description; final String? description;
final String? imageUrl; final String? imageUrl;
final bool? isLiked; final bool isLiked;
final String? id;
final String? name; final String? name;
final String? surname; final String? surname;
final OnLikeCallback onLike; final OnLikeCallback onLike;
final VoidCallback? onTap; final VoidCallback? onTap;
const CardPost( const _Card(
{this.name, {this.name,
this.surname, this.surname,
this.description, this.description,
this.imageUrl, this.imageUrl,
this.isLiked, this.isLiked = false,
this.onLike, this.onLike,
this.onTap}); this.onTap,
this.id});
factory CardPost.fromData(CardPostData data, factory _Card.fromData(CardData data,
{OnLikeCallback onLike, VoidCallback? onTap}) => {OnLikeCallback onLike, VoidCallback? onTap, bool isLiked = false}) =>
CardPost( _Card(
name: data.name, name: data.name,
surname: data.surname, surname: data.surname,
description: data.description, description: data.description,
imageUrl: data.imageUrl, imageUrl: data.imageUrl,
isLiked: data.isLiked, isLiked: isLiked,
onLike: onLike, onLike: onLike,
onTap: onTap); onTap: onTap,
id: data.id,
@override );
State<CardPost> createState() => _CardPostState();
}
class _CardPostState extends State<CardPost> {
bool isLiked = false;
double iconScale = NORMAL_ICON_SCALE;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return GestureDetector( return GestureDetector(
onTap: widget.onTap, onTap: onTap,
child: Container( child: Container(
margin: const EdgeInsets.all(10), margin: const EdgeInsets.all(10),
padding: const EdgeInsets.all(10), padding: const EdgeInsets.all(10),
@@ -80,7 +72,7 @@ class _CardPostState extends State<CardPost> {
children: [ children: [
Flexible( Flexible(
child: Text( child: Text(
(widget.name ?? "") + (" ") + (widget.surname ?? ""), (name ?? "") + (" ") + (surname ?? ""),
style: const TextStyle( style: const TextStyle(
color: Colors.white, color: Colors.white,
fontSize: 30, fontSize: 30,
@@ -96,7 +88,7 @@ class _CardPostState extends State<CardPost> {
children: [ children: [
Flexible( Flexible(
child: Text( child: Text(
widget.description ?? "", description ?? "",
textAlign: TextAlign.center, textAlign: TextAlign.center,
style: const TextStyle( style: const TextStyle(
color: Colors.white, color: Colors.white,
@@ -107,50 +99,40 @@ class _CardPostState extends State<CardPost> {
], ],
), ),
), ),
if (widget.imageUrl != null) if (imageUrl != null)
Padding( Padding(
padding: const EdgeInsets.all(8.0), padding: const EdgeInsets.all(8.0),
child: Image.network(widget.imageUrl!, child: Image.network(imageUrl!,
height: 500, height: 500,
width: double.infinity, width: double.infinity,
fit: BoxFit.fitWidth), fit: BoxFit.fitWidth),
), ),
Padding( Padding(
padding: const EdgeInsets.all(4.0), padding: const EdgeInsets.all(4.0),
child: Row( child: Row(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
GestureDetector( GestureDetector(
onTap: () => { onTap: () => onLike?.call(id, "${name!} ${surname!}", isLiked),
setState(() { child: AnimatedSwitcher(
isLiked = !isLiked; duration: const Duration(milliseconds: 250),
iconScale = SCALED_ICON_SCALE; child: !isLiked
? const Icon(
Timer(const Duration(milliseconds: 110), () { Icons.favorite_border,
setState(() { color: Colors.black,
iconScale = NORMAL_ICON_SCALE; key: ValueKey<int>(0),
}); )
}); : const Icon(
widget.onLike?.call(widget.name ?? "", isLiked); Icons.favorite,
}) color: Colors.red,
}, key: ValueKey<int>(1),
child: AnimatedScale( ),
scale: iconScale, ),
duration: const Duration(milliseconds: 250),
child: AnimatedSwitcher(
duration: const Duration(milliseconds: 250),
child: isLiked
? const Icon(
Icons.favorite,
color: Colors.red,
key: ValueKey(1),
)
: const Icon(Icons.favorite_border,
color: Colors.black, key: ValueKey(1)))),
), ),
], ],
), ),
) ),
], ],
), ),
)); ));

View File

@@ -1,36 +1,197 @@
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:pmu/data/repositories/api_user_repository.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:pmu/domain/card.dart'; import 'package:pmu/components/debounce.dart';
import 'package:pmu/components/extensions/content_x.dart';
import 'package:pmu/domain/models/card.dart';
import 'package:pmu/presentation/common/svg_objects.dart';
import 'package:pmu/presentation/detail_pages/card_detail_page.dart'; import 'package:pmu/presentation/detail_pages/card_detail_page.dart';
import 'package:pmu/presentation/home_page/card.dart'; import 'package:pmu/presentation/home_page/bloc/bloc.dart';
import 'package:pmu/presentation/home_page/bloc/events.dart';
import 'package:pmu/presentation/home_page/bloc/state.dart';
import 'package:pmu/presentation/locale/bloc.dart';
import 'package:pmu/presentation/locale/events.dart';
import 'package:pmu/presentation/locale/state.dart';
class MyHomePage extends StatefulWidget { import '../like/bloc.dart';
const MyHomePage({super.key, required this.title}); import '../like/events.dart';
import '../like/state.dart';
final String title; part 'card.dart';
class HomePage extends StatefulWidget {
const HomePage({super.key});
@override @override
State<MyHomePage> createState() => _MyHomePageState(); State<HomePage> createState() => _HomePageState();
} }
class _MyHomePageState extends State<MyHomePage> { class _HomePageState extends State<HomePage> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Scaffold( return const Scaffold(body: _Body());
appBar: AppBar(
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
title: Text(widget.title),
),
body: const Body(),
);
} }
} }
class Body extends StatelessWidget { class _Body extends StatefulWidget {
const Body({super.key}); const _Body();
void _navToDetails(BuildContext context, CardPostData data) { @override
State<StatefulWidget> createState() => _BodyState();
}
class _BodyState extends State<_Body> {
final searchController = TextEditingController();
final scrollController = ScrollController();
@override
void initState() {
WidgetsBinding.instance.addPostFrameCallback((_) {
context.read<HomeBloc>().add(const HomeLoadDataEvent());
context.read<LikeBloc>().add(const LoadLikesEvent());
});
scrollController.addListener(_onNextPageListener);
super.initState();
}
void _onNextPageListener() {
if (scrollController.offset >= scrollController.position.maxScrollExtent) {
// preventing multiple pagination request on multiple swipes
final bloc = context.read<HomeBloc>();
if (!bloc.state.isPaginationLoading) {
bloc.add(HomeLoadDataEvent(
search: searchController.text,
pageNumber: bloc.state.data?.pageNumber,
nextPageNumber: bloc.state.data?.nextPageNumber,
lastPageNumber: bloc.state.data?.lastPageNumber
));
}
}
}
@override
void dispose() {
searchController.dispose();
scrollController.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
return Padding(
padding: EdgeInsets.only(top: MediaQuery
.of(context)
.padding
.top),
child: Column(
children: [
Row(
children: [
Expanded(
flex: 4,
child: Padding(
padding: const EdgeInsets.all(8.0),
child: CupertinoSearchTextField(
controller: searchController,
placeholder: context.locale.search,
onChanged: (search) {
Debounce.run(() =>
context
.read<HomeBloc>()
.add(HomeLoadDataEvent(search: search)));
},
),
),
),
GestureDetector(
onTap: () =>
context.read<LocaleBloc>().add(const ChangeLocaleEvent()),
child: SizedBox.square(
dimension: 50,
child: Padding(
padding: const EdgeInsets.all(12),
child: BlocBuilder<LocaleBloc, LocaleState>(
builder: (context, state) {
return state.currentLocale.languageCode == 'ru'
? const SvgRu()
: const SvgUk();
},
),
),
),
)
],
),
BlocBuilder<HomeBloc, HomeState>(
builder: (context, state) =>
state.error != null
? Text(
state.error ?? '',
style: Theme
.of(context)
.textTheme
.headlineSmall
?.copyWith(color: Colors.red),
)
: state.isLoading
? const CircularProgressIndicator()
: BlocBuilder<LikeBloc, LikeState>(
builder: (context, likeState) {
return Expanded(
child: RefreshIndicator(
onRefresh: _onRefresh,
child: ListView.builder(
controller: scrollController,
padding: EdgeInsets.zero,
itemCount: state.data?.data?.length ?? 0,
itemBuilder: (context, index) {
final data = state.data?.data?[index];
return data != null
? _Card.fromData(
data,
onLike: _onLike,
isLiked: likeState.likedIds
?.contains(data.id) ==
true,
onTap: () =>
_navToDetails(context, data),
)
: const SizedBox.shrink();
},
),
),
);
},
),
),
BlocBuilder<HomeBloc, HomeState>(
builder: (context, state) =>
state.isPaginationLoading
? const CircularProgressIndicator()
: const SizedBox.shrink(),
),
],
),
);
}
void _onLike(String? id, String title, bool isLiked) {
if (id != null) {
context.read<LikeBloc>().add(ChangeLikeEvent(id));
_showSnackBar(context, title, !isLiked);
}
}
Future<void> _onRefresh() {
context
.read<HomeBloc>()
.add(HomeLoadDataEvent(search: searchController.text));
return Future.value(null);
}
void _navToDetails(BuildContext context, CardData data) {
Navigator.push( Navigator.push(
context, context,
CupertinoPageRoute(builder: (context) => DetailPage(data)), CupertinoPageRoute(builder: (context) => DetailPage(data)),
@@ -41,7 +202,7 @@ class Body extends StatelessWidget {
WidgetsBinding.instance.addPostFrameCallback((_) { WidgetsBinding.instance.addPostFrameCallback((_) {
ScaffoldMessenger.of(context).showSnackBar(SnackBar( ScaffoldMessenger.of(context).showSnackBar(SnackBar(
content: Text( content: Text(
'${isLiked ? 'Вы поставили лайк: ' : 'Вы убрали лайк: '} $title', '${isLiked ? context.locale.liked : context.locale.disliked } $title',
style: const TextStyle( style: const TextStyle(
color: Colors.black45, fontSize: 20, fontWeight: FontWeight.bold), color: Colors.black45, fontSize: 20, fontWeight: FontWeight.bold),
), ),
@@ -50,27 +211,4 @@ class Body extends StatelessWidget {
)); ));
}); });
} }
@override
Widget build(BuildContext context) {
final data = ApiUserRepository().loadData();
return Center(
child: FutureBuilder<List<CardPostData>?>(
future: data,
builder: (context, snapshot) => SingleChildScrollView(
child: snapshot.hasData
? Column(
mainAxisAlignment: MainAxisAlignment.center,
children: snapshot.data
?.map((data) => CardPost.fromData(data,
onLike: (String title, bool isLiked) =>
_showSnackBar(context, title, isLiked),
onTap: () => _navToDetails(context, data)))
.toList() ??
[])
: const CircularProgressIndicator(),
),
),
);
}
} }

View File

@@ -0,0 +1,38 @@
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:pmu/presentation/like/events.dart';
import 'package:pmu/presentation/like/state.dart';
import 'package:shared_preferences/shared_preferences.dart';
const String _likedPrefsKey = 'liked';
class LikeBloc extends Bloc<LikeEvent, LikeState> {
LikeBloc() : super(const LikeState(likedIds: [])) {
on<ChangeLikeEvent>(_onChangeLike);
on<LoadLikesEvent>(_onLoadLikes);
}
Future<void> _onLoadLikes(
LoadLikesEvent event, Emitter<LikeState> emit) async {
final prefs = await SharedPreferences.getInstance();
final data = prefs.getStringList(_likedPrefsKey);
emit(state.copyWith(likedIds: data));
}
Future<void> _onChangeLike(
ChangeLikeEvent event, Emitter<LikeState> emit) async {
final updatedList = List<String>.from(state.likedIds ?? []);
if (updatedList.contains(event.id)) {
updatedList.remove(event.id);
} else {
updatedList.add(event.id);
}
final prefs = await SharedPreferences.getInstance();
prefs.setStringList(_likedPrefsKey, updatedList);
emit(state.copyWith(likedIds: updatedList));
}
}

View File

@@ -0,0 +1,13 @@
abstract class LikeEvent {
const LikeEvent();
}
class LoadLikesEvent extends LikeEvent {
const LoadLikesEvent();
}
class ChangeLikeEvent extends LikeEvent {
final String id;
const ChangeLikeEvent(this.id);
}

View File

@@ -0,0 +1,14 @@
import 'package:copy_with_extension/copy_with_extension.dart';
import 'package:equatable/equatable.dart';
part 'state.g.dart';
@CopyWith()
class LikeState extends Equatable {
final List<String>? likedIds;
const LikeState({required this.likedIds});
@override
List<Object?> get props => [likedIds];
}

View File

@@ -0,0 +1,56 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'state.dart';
// **************************************************************************
// CopyWithGenerator
// **************************************************************************
abstract class _$LikeStateCWProxy {
LikeState likedIds(List<String>? likedIds);
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `LikeState(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// LikeState(...).copyWith(id: 12, name: "My name")
/// ````
LikeState call({
List<String>? likedIds,
});
}
/// Proxy class for `copyWith` functionality. This is a callable class and can be used as follows: `instanceOfLikeState.copyWith(...)`. Additionally contains functions for specific fields e.g. `instanceOfLikeState.copyWith.fieldName(...)`
class _$LikeStateCWProxyImpl implements _$LikeStateCWProxy {
const _$LikeStateCWProxyImpl(this._value);
final LikeState _value;
@override
LikeState likedIds(List<String>? likedIds) => this(likedIds: likedIds);
@override
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `LikeState(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// LikeState(...).copyWith(id: 12, name: "My name")
/// ````
LikeState call({
Object? likedIds = const $CopyWithPlaceholder(),
}) {
return LikeState(
likedIds: likedIds == const $CopyWithPlaceholder()
? _value.likedIds
// ignore: cast_nullable_to_non_nullable
: likedIds as List<String>?,
);
}
}
extension $LikeStateCopyWith on LikeState {
/// Returns a callable class that can be used as follows: `instanceOfLikeState.copyWith(...)` or like so:`instanceOfLikeState.copyWith.fieldName(...)`.
// ignore: library_private_types_in_public_api
_$LikeStateCWProxy get copyWith => _$LikeStateCWProxyImpl(this);
}

View File

@@ -0,0 +1,19 @@
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:pmu/components/locale/l10n/app_locale.dart';
import 'package:pmu/presentation/locale/events.dart';
import 'package:pmu/presentation/locale/state.dart';
class LocaleBloc extends Bloc<LocaleEvent, LocaleState> {
LocaleBloc(Locale defaultLocale)
: super(LocaleState(currentLocale: defaultLocale)) {
on<ChangeLocaleEvent>(_onChangeLocale);
}
Future<void> _onChangeLocale(
ChangeLocaleEvent event, Emitter<LocaleState> emit) async {
final toChange = AppLocale.supportedLocales
.firstWhere((e) => e.languageCode != state.currentLocale.languageCode);
emit(state.copyWith(currentLocale: toChange));
}
}

View File

@@ -0,0 +1,7 @@
abstract class LocaleEvent {
const LocaleEvent();
}
class ChangeLocaleEvent extends LocaleEvent {
const ChangeLocaleEvent();
}

View File

@@ -0,0 +1,16 @@
import 'dart:ui';
import 'package:copy_with_extension/copy_with_extension.dart';
import 'package:equatable/equatable.dart';
part 'state.g.dart';
@CopyWith()
class LocaleState extends Equatable {
final Locale currentLocale;
const LocaleState({required this.currentLocale});
@override
List<Object?> get props => [currentLocale];
}

View File

@@ -0,0 +1,57 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'state.dart';
// **************************************************************************
// CopyWithGenerator
// **************************************************************************
abstract class _$LocaleStateCWProxy {
LocaleState currentLocale(Locale currentLocale);
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `LocaleState(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// LocaleState(...).copyWith(id: 12, name: "My name")
/// ````
LocaleState call({
Locale currentLocale,
});
}
/// Proxy class for `copyWith` functionality. This is a callable class and can be used as follows: `instanceOfLocaleState.copyWith(...)`. Additionally contains functions for specific fields e.g. `instanceOfLocaleState.copyWith.fieldName(...)`
class _$LocaleStateCWProxyImpl implements _$LocaleStateCWProxy {
const _$LocaleStateCWProxyImpl(this._value);
final LocaleState _value;
@override
LocaleState currentLocale(Locale currentLocale) =>
this(currentLocale: currentLocale);
@override
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `LocaleState(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// LocaleState(...).copyWith(id: 12, name: "My name")
/// ````
LocaleState call({
Object? currentLocale = const $CopyWithPlaceholder(),
}) {
return LocaleState(
currentLocale: currentLocale == const $CopyWithPlaceholder()
? _value.currentLocale
// ignore: cast_nullable_to_non_nullable
: currentLocale as Locale,
);
}
}
extension $LocaleStateCopyWith on LocaleState {
/// Returns a callable class that can be used as follows: `instanceOfLocaleState.copyWith(...)` or like so:`instanceOfLocaleState.copyWith.fieldName(...)`.
// ignore: library_private_types_in_public_api
_$LocaleStateCWProxy get copyWith => _$LocaleStateCWProxyImpl(this);
}

View File

@@ -22,6 +22,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "6.7.0" version: "6.7.0"
archive:
dependency: transitive
description:
name: archive
sha256: "08064924cbf0ab88280a0c3f60db9dd24fec693927e725ecb176f16c629d1cb8"
url: "https://pub.dev"
source: hosted
version: "4.0.1"
args: args:
dependency: transitive dependency: transitive
description: description:
@@ -38,6 +46,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "2.11.0" version: "2.11.0"
bloc:
dependency: transitive
description:
name: bloc
sha256: "106842ad6569f0b60297619e9e0b1885c2fb9bf84812935490e6c5275777804e"
url: "https://pub.dev"
source: hosted
version: "8.1.4"
boolean_selector: boolean_selector:
dependency: transitive dependency: transitive
description: description:
@@ -126,6 +142,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "2.0.3" version: "2.0.3"
cli_util:
dependency: transitive
description:
name: cli_util
sha256: ff6785f7e9e3c38ac98b2fb035701789de90154024a75b6cb926445e83197d1c
url: "https://pub.dev"
source: hosted
version: "0.4.2"
clock: clock:
dependency: transitive dependency: transitive
description: description:
@@ -158,6 +182,22 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "3.1.2" version: "3.1.2"
copy_with_extension:
dependency: "direct main"
description:
name: copy_with_extension
sha256: ed472ae80d807094d7a7d7ef67901f8167d18c7998e6db81785a51364aede627
url: "https://pub.dev"
source: hosted
version: "6.0.0"
copy_with_extension_gen:
dependency: "direct main"
description:
name: copy_with_extension_gen
sha256: "0be2694d3d50df16d91c04e7444181bfb2a0ad8a3b169d58de0c38a69864e4bd"
url: "https://pub.dev"
source: hosted
version: "6.0.0"
crypto: crypto:
dependency: transitive dependency: transitive
description: description:
@@ -198,6 +238,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "2.0.0" version: "2.0.0"
equatable:
dependency: "direct main"
description:
name: equatable
sha256: "567c64b3cb4cf82397aac55f4f0cbd3ca20d77c6c03bedbc4ceaddc08904aef7"
url: "https://pub.dev"
source: hosted
version: "2.0.7"
fake_async: fake_async:
dependency: transitive dependency: transitive
description: description:
@@ -206,6 +254,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.3.1" version: "1.3.1"
ffi:
dependency: transitive
description:
name: ffi
sha256: "16ed7b077ef01ad6170a3d0c57caa4a112a38d7a2ed5602e0aca9ca6f3d98da6"
url: "https://pub.dev"
source: hosted
version: "2.1.3"
file: file:
dependency: transitive dependency: transitive
description: description:
@@ -227,6 +283,22 @@ packages:
description: flutter description: flutter
source: sdk source: sdk
version: "0.0.0" version: "0.0.0"
flutter_bloc:
dependency: "direct main"
description:
name: flutter_bloc
sha256: b594505eac31a0518bdcb4b5b79573b8d9117b193cc80cc12e17d639b10aa27a
url: "https://pub.dev"
source: hosted
version: "8.1.6"
flutter_launcher_icons:
dependency: "direct main"
description:
name: flutter_launcher_icons
sha256: "526faf84284b86a4cb36d20a5e45147747b7563d921373d4ee0559c54fcdbcea"
url: "https://pub.dev"
source: hosted
version: "0.13.1"
flutter_lints: flutter_lints:
dependency: "direct dev" dependency: "direct dev"
description: description:
@@ -235,11 +307,29 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "4.0.0" version: "4.0.0"
flutter_localizations:
dependency: "direct main"
description: flutter
source: sdk
version: "0.0.0"
flutter_svg:
dependency: "direct main"
description:
name: flutter_svg
sha256: "8c5d68a82add3ca76d792f058b186a0599414f279f00ece4830b9b231b570338"
url: "https://pub.dev"
source: hosted
version: "2.0.7"
flutter_test: flutter_test:
dependency: "direct dev" dependency: "direct dev"
description: flutter description: flutter
source: sdk source: sdk
version: "0.0.0" version: "0.0.0"
flutter_web_plugins:
dependency: transitive
description: flutter
source: sdk
version: "0.0.0"
frontend_server_client: frontend_server_client:
dependency: transitive dependency: transitive
description: description:
@@ -264,6 +354,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "2.3.2" version: "2.3.2"
http:
dependency: transitive
description:
name: http
sha256: b9c29a161230ee03d3ccf545097fccd9b87a5264228c5d348202e0f0c28f9010
url: "https://pub.dev"
source: hosted
version: "1.2.2"
http_multi_server: http_multi_server:
dependency: transitive dependency: transitive
description: description:
@@ -280,6 +378,22 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "4.0.2" version: "4.0.2"
image:
dependency: transitive
description:
name: image
sha256: b50b415345578583de0f1cf4c7bd389f164de0b316d890c707b41133047dbc2a
url: "https://pub.dev"
source: hosted
version: "4.5.1"
intl:
dependency: "direct main"
description:
name: intl
sha256: d6f56758b7d3014a48af9701c085700aac781a92a87a62b1333b46d8879661cf
url: "https://pub.dev"
source: hosted
version: "0.19.0"
io: io:
dependency: transitive dependency: transitive
description: description:
@@ -392,6 +506,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "2.0.0" version: "2.0.0"
nested:
dependency: transitive
description:
name: nested
sha256: "03bac4c528c64c95c722ec99280375a6f2fc708eec17c7b3f07253b626cd2a20"
url: "https://pub.dev"
source: hosted
version: "1.0.0"
package_config: package_config:
dependency: transitive dependency: transitive
description: description:
@@ -408,6 +530,62 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.9.0" version: "1.9.0"
path_parsing:
dependency: transitive
description:
name: path_parsing
sha256: "883402936929eac138ee0a45da5b0f2c80f89913e6dc3bf77eb65b84b409c6ca"
url: "https://pub.dev"
source: hosted
version: "1.1.0"
path_provider_linux:
dependency: transitive
description:
name: path_provider_linux
sha256: f7a1fe3a634fe7734c8d3f2766ad746ae2a2884abe22e241a8b301bf5cac3279
url: "https://pub.dev"
source: hosted
version: "2.2.1"
path_provider_platform_interface:
dependency: transitive
description:
name: path_provider_platform_interface
sha256: "88f5779f72ba699763fa3a3b06aa4bf6de76c8e5de842cf6f29e2e06476c2334"
url: "https://pub.dev"
source: hosted
version: "2.1.2"
path_provider_windows:
dependency: transitive
description:
name: path_provider_windows
sha256: bd6f00dbd873bfb70d0761682da2b3a2c2fccc2b9e84c495821639601d81afe7
url: "https://pub.dev"
source: hosted
version: "2.3.0"
petitparser:
dependency: transitive
description:
name: petitparser
sha256: c15605cd28af66339f8eb6fbe0e541bfe2d1b72d5825efc6598f3e0a31b9ad27
url: "https://pub.dev"
source: hosted
version: "6.0.2"
platform:
dependency: transitive
description:
name: platform
sha256: "5d6b1b0036a5f331ebc77c850ebc8506cbc1e9416c27e59b439f917a902a4984"
url: "https://pub.dev"
source: hosted
version: "3.1.6"
plugin_platform_interface:
dependency: transitive
description:
name: plugin_platform_interface
sha256: "4820fbfdb9478b1ebae27888254d445073732dae3d6ea81f0b7e06d5dedc3f02"
url: "https://pub.dev"
source: hosted
version: "2.1.8"
pool: pool:
dependency: transitive dependency: transitive
description: description:
@@ -416,6 +594,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.5.1" version: "1.5.1"
posix:
dependency: transitive
description:
name: posix
sha256: a0117dc2167805aa9125b82eee515cc891819bac2f538c83646d355b16f58b9a
url: "https://pub.dev"
source: hosted
version: "6.0.1"
pretty_dio_logger: pretty_dio_logger:
dependency: "direct main" dependency: "direct main"
description: description:
@@ -424,6 +610,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.4.0" version: "1.4.0"
provider:
dependency: transitive
description:
name: provider
sha256: c8a055ee5ce3fd98d6fc872478b03823ffdb448699c6ebdbbc71d59b596fd48c
url: "https://pub.dev"
source: hosted
version: "6.1.2"
pub_semver: pub_semver:
dependency: transitive dependency: transitive
description: description:
@@ -440,6 +634,62 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.3.0" version: "1.3.0"
shared_preferences:
dependency: "direct main"
description:
name: shared_preferences
sha256: d3bbe5553a986e83980916ded2f0b435ef2e1893dfaa29d5a7a790d0eca12180
url: "https://pub.dev"
source: hosted
version: "2.2.3"
shared_preferences_android:
dependency: transitive
description:
name: shared_preferences_android
sha256: "02a7d8a9ef346c9af715811b01fbd8e27845ad2c41148eefd31321471b41863d"
url: "https://pub.dev"
source: hosted
version: "2.4.0"
shared_preferences_foundation:
dependency: transitive
description:
name: shared_preferences_foundation
sha256: "6a52cfcdaeac77cad8c97b539ff688ccfc458c007b4db12be584fbe5c0e49e03"
url: "https://pub.dev"
source: hosted
version: "2.5.4"
shared_preferences_linux:
dependency: transitive
description:
name: shared_preferences_linux
sha256: "580abfd40f415611503cae30adf626e6656dfb2f0cee8f465ece7b6defb40f2f"
url: "https://pub.dev"
source: hosted
version: "2.4.1"
shared_preferences_platform_interface:
dependency: transitive
description:
name: shared_preferences_platform_interface
sha256: "57cbf196c486bc2cf1f02b85784932c6094376284b3ad5779d1b1c6c6a816b80"
url: "https://pub.dev"
source: hosted
version: "2.4.1"
shared_preferences_web:
dependency: transitive
description:
name: shared_preferences_web
sha256: d2ca4132d3946fec2184261726b355836a82c33d7d5b67af32692aff18a4684e
url: "https://pub.dev"
source: hosted
version: "2.4.2"
shared_preferences_windows:
dependency: transitive
description:
name: shared_preferences_windows
sha256: "94ef0f72b2d71bc3e700e025db3710911bd51a71cefb65cc609dd0d9a982e3c1"
url: "https://pub.dev"
source: hosted
version: "2.4.1"
shelf: shelf:
dependency: transitive dependency: transitive
description: description:
@@ -549,6 +799,30 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.4.0" version: "1.4.0"
vector_graphics:
dependency: transitive
description:
name: vector_graphics
sha256: "27d5fefe86fb9aace4a9f8375b56b3c292b64d8c04510df230f849850d912cb7"
url: "https://pub.dev"
source: hosted
version: "1.1.15"
vector_graphics_codec:
dependency: transitive
description:
name: vector_graphics_codec
sha256: "2430b973a4ca3c4dbc9999b62b8c719a160100dcbae5c819bae0cacce32c9cdb"
url: "https://pub.dev"
source: hosted
version: "1.1.12"
vector_graphics_compiler:
dependency: transitive
description:
name: vector_graphics_compiler
sha256: "1b4b9e706a10294258727674a340ae0d6e64a7231980f9f9a3d12e4b42407aad"
url: "https://pub.dev"
source: hosted
version: "1.1.16"
vector_math: vector_math:
dependency: transitive dependency: transitive
description: description:
@@ -597,6 +871,22 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "3.0.1" version: "3.0.1"
xdg_directories:
dependency: transitive
description:
name: xdg_directories
sha256: "7a3f37b05d989967cdddcbb571f1ea834867ae2faa29725fd085180e0883aa15"
url: "https://pub.dev"
source: hosted
version: "1.1.0"
xml:
dependency: transitive
description:
name: xml
sha256: b015a8ad1c488f66851d762d3090a21c600e479dc75e68328c52774040cf9226
url: "https://pub.dev"
source: hosted
version: "6.5.0"
yaml: yaml:
dependency: transitive dependency: transitive
description: description:
@@ -607,4 +897,4 @@ packages:
version: "3.1.2" version: "3.1.2"
sdks: sdks:
dart: ">=3.5.3 <4.0.0" dart: ">=3.5.3 <4.0.0"
flutter: ">=3.18.0-18.0.pre.54" flutter: ">=3.24.0"

View File

@@ -34,11 +34,32 @@ dependencies:
dio: ^5.4.2+1 dio: ^5.4.2+1
pretty_dio_logger: ^1.3.1 pretty_dio_logger: ^1.3.1
json_annotation: ^4.8.1 json_annotation: ^4.8.1
flutter_svg: 2.0.7
# Localization
flutter_localizations:
sdk: flutter
intl: 0.19.0
shared_preferences: 2.2.3
# Иконки
flutter_launcher_icons: 0.13.1
copy_with_extension: ^6.0.0
copy_with_extension_gen: ^6.0.0
# The following adds the Cupertino Icons font to your application. # The following adds the Cupertino Icons font to your application.
# Use with the CupertinoIcons class for iOS style icons. # Use with the CupertinoIcons class for iOS style icons.
cupertino_icons: ^1.0.8 cupertino_icons: ^1.0.8
flutter_bloc: ^8.1.6
equatable: ^2.0.7
flutter_icons:
android: "ic_launcher"
image_path: "assets/launcher.jpg"
min_sdk_android: 21
dev_dependencies: dev_dependencies:
flutter_test: flutter_test:
sdk: flutter sdk: flutter
@@ -58,12 +79,16 @@ dev_dependencies:
# The following section is specific to Flutter packages. # The following section is specific to Flutter packages.
flutter: flutter:
generate: true
# The following line ensures that the Material Icons font is # The following line ensures that the Material Icons font is
# included with your application, so that you can use the icons in # included with your application, so that you can use the icons in
# the material Icons class. # the material Icons class.
uses-material-design: true uses-material-design: true
assets:
- assets/svg/
# To add assets to your application, add an assets section, like this: # To add assets to your application, add an assets section, like this:
# assets: # assets:
# - images/a_dot_burr.jpeg # - images/a_dot_burr.jpeg