add little bit of localization, add SharedPreferences for likes, add change locale button
This commit is contained in:
parent
9cb990d7ac
commit
011fbbd05f
@ -42,4 +42,5 @@
|
||||
<data android:mimeType="text/plain"/>
|
||||
</intent>
|
||||
</queries>
|
||||
<uses-permission android:name="android.permission.INTERNET"/>
|
||||
</manifest>
|
||||
|
@ -1,12 +1,21 @@
|
||||
{
|
||||
"@@locale": "en",
|
||||
|
||||
"appBarTitle": "Anime List",
|
||||
|
||||
"search": "Search",
|
||||
"liked": "You liked",
|
||||
"unliked": "Like removed from",
|
||||
"errorOccured": "Error occured",
|
||||
"noErrorMsg": "No message provided",
|
||||
"retry": "Retry",
|
||||
"unknown": "Unknown",
|
||||
|
||||
"apiYear": "Year",
|
||||
"apiType": "Type",
|
||||
"apiRating": "Rating",
|
||||
"apiDesc": "",
|
||||
"apiNoDesc": "No description provided",
|
||||
|
||||
"arbEnding": "t"
|
||||
}
|
@ -1,12 +1,21 @@
|
||||
{
|
||||
"@@locale": "ru",
|
||||
|
||||
"appBarTitle": "Список аниме",
|
||||
|
||||
"search": "Поиск",
|
||||
"liked": "Вы лайкнули",
|
||||
"unliked": "Лайк снят с",
|
||||
"errorOccured": "Произошла ошибка",
|
||||
"noErrorMsg": "Нет сообщения",
|
||||
"retry": "Повторить",
|
||||
"unknown": "Неизвестно",
|
||||
|
||||
"apiYear": "Год",
|
||||
"apiType": "Тип",
|
||||
"apiRating": "Рейтинг",
|
||||
"apiDesc": "(Описание доступно только на английском языке)",
|
||||
"apiNoDesc": "Нет описания",
|
||||
|
||||
"arbEnding": "t"
|
||||
}
|
@ -95,6 +95,12 @@ abstract class AppLocale {
|
||||
Locale('ru')
|
||||
];
|
||||
|
||||
/// No description provided for @appBarTitle.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
/// **'Список аниме'**
|
||||
String get appBarTitle;
|
||||
|
||||
/// No description provided for @search.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
@ -131,6 +137,42 @@ abstract class AppLocale {
|
||||
/// **'Повторить'**
|
||||
String get retry;
|
||||
|
||||
/// No description provided for @unknown.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
/// **'Неизвестно'**
|
||||
String get unknown;
|
||||
|
||||
/// No description provided for @apiYear.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
/// **'Год'**
|
||||
String get apiYear;
|
||||
|
||||
/// No description provided for @apiType.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
/// **'Тип'**
|
||||
String get apiType;
|
||||
|
||||
/// No description provided for @apiRating.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
/// **'Рейтинг'**
|
||||
String get apiRating;
|
||||
|
||||
/// No description provided for @apiDesc.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
/// **'(Описание доступно только на английском языке)'**
|
||||
String get apiDesc;
|
||||
|
||||
/// No description provided for @apiNoDesc.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
/// **'Нет описания'**
|
||||
String get apiNoDesc;
|
||||
|
||||
/// No description provided for @arbEnding.
|
||||
///
|
||||
/// In ru, this message translates to:
|
||||
|
@ -6,6 +6,9 @@ import 'app_localizations.dart';
|
||||
class AppLocaleEn extends AppLocale {
|
||||
AppLocaleEn([String locale = 'en']) : super(locale);
|
||||
|
||||
@override
|
||||
String get appBarTitle => 'Anime List';
|
||||
|
||||
@override
|
||||
String get search => 'Search';
|
||||
|
||||
@ -24,6 +27,24 @@ class AppLocaleEn extends AppLocale {
|
||||
@override
|
||||
String get retry => 'Retry';
|
||||
|
||||
@override
|
||||
String get unknown => 'Unknown';
|
||||
|
||||
@override
|
||||
String get apiYear => 'Year';
|
||||
|
||||
@override
|
||||
String get apiType => 'Type';
|
||||
|
||||
@override
|
||||
String get apiRating => 'Rating';
|
||||
|
||||
@override
|
||||
String get apiDesc => '';
|
||||
|
||||
@override
|
||||
String get apiNoDesc => 'No description provided';
|
||||
|
||||
@override
|
||||
String get arbEnding => 't';
|
||||
}
|
||||
|
@ -6,6 +6,9 @@ import 'app_localizations.dart';
|
||||
class AppLocaleRu extends AppLocale {
|
||||
AppLocaleRu([String locale = 'ru']) : super(locale);
|
||||
|
||||
@override
|
||||
String get appBarTitle => 'Список аниме';
|
||||
|
||||
@override
|
||||
String get search => 'Поиск';
|
||||
|
||||
@ -24,6 +27,24 @@ class AppLocaleRu extends AppLocale {
|
||||
@override
|
||||
String get retry => 'Повторить';
|
||||
|
||||
@override
|
||||
String get unknown => 'Неизвестно';
|
||||
|
||||
@override
|
||||
String get apiYear => 'Год';
|
||||
|
||||
@override
|
||||
String get apiType => 'Тип';
|
||||
|
||||
@override
|
||||
String get apiRating => 'Рейтинг';
|
||||
|
||||
@override
|
||||
String get apiDesc => '(Описание доступно только на английском языке)';
|
||||
|
||||
@override
|
||||
String get apiNoDesc => 'Нет описания';
|
||||
|
||||
@override
|
||||
String get arbEnding => 't';
|
||||
}
|
||||
|
@ -8,7 +8,8 @@ class AnimesDto {
|
||||
final PaginationDto? pagination;
|
||||
const AnimesDto({this.data, this.pagination});
|
||||
|
||||
factory AnimesDto.fromJson(Map<String, dynamic> json) => _$AnimesDtoFromJson(json);
|
||||
factory AnimesDto.fromJson(Map<String, dynamic> json) =>
|
||||
_$AnimesDtoFromJson(json);
|
||||
}
|
||||
|
||||
@JsonSerializable(createToJson: false)
|
||||
@ -22,11 +23,14 @@ class PaginationDto {
|
||||
|
||||
const PaginationDto({this.currentPage, this.hasNextPage, this.lastPage});
|
||||
|
||||
factory PaginationDto.fromJson(Map<String, dynamic> json) => _$PaginationDtoFromJson(json);
|
||||
factory PaginationDto.fromJson(Map<String, dynamic> json) =>
|
||||
_$PaginationDtoFromJson(json);
|
||||
}
|
||||
|
||||
@JsonSerializable(createToJson: false)
|
||||
class AnimeDto {
|
||||
@JsonKey(name: "mal_id")
|
||||
final int? id;
|
||||
final String? title;
|
||||
final int? year;
|
||||
final String? type;
|
||||
@ -34,9 +38,17 @@ class AnimeDto {
|
||||
final String? rating;
|
||||
final ImagesDto? images;
|
||||
|
||||
const AnimeDto({this.title, this.rating, this.synopsis, this.type, this.year, this.images});
|
||||
const AnimeDto(
|
||||
{this.id,
|
||||
this.title,
|
||||
this.rating,
|
||||
this.synopsis,
|
||||
this.type,
|
||||
this.year,
|
||||
this.images});
|
||||
|
||||
factory AnimeDto.fromJson(Map<String, dynamic> json) => _$AnimeDtoFromJson(json);
|
||||
factory AnimeDto.fromJson(Map<String, dynamic> json) =>
|
||||
_$AnimeDtoFromJson(json);
|
||||
}
|
||||
|
||||
@JsonSerializable(createToJson: false)
|
||||
@ -45,7 +57,8 @@ class ImagesDto {
|
||||
|
||||
const ImagesDto({this.jpg});
|
||||
|
||||
factory ImagesDto.fromJson(Map<String, dynamic> json) => _$ImagesDtoFromJson(json);
|
||||
factory ImagesDto.fromJson(Map<String, dynamic> json) =>
|
||||
_$ImagesDtoFromJson(json);
|
||||
}
|
||||
|
||||
@JsonSerializable(createToJson: false)
|
||||
@ -55,5 +68,6 @@ class ImageDto {
|
||||
|
||||
const ImageDto({this.imageUrl});
|
||||
|
||||
factory ImageDto.fromJson(Map<String, dynamic> json) => _$ImageDtoFromJson(json);
|
||||
factory ImageDto.fromJson(Map<String, dynamic> json) =>
|
||||
_$ImageDtoFromJson(json);
|
||||
}
|
||||
|
@ -23,6 +23,7 @@ PaginationDto _$PaginationDtoFromJson(Map<String, dynamic> json) =>
|
||||
);
|
||||
|
||||
AnimeDto _$AnimeDtoFromJson(Map<String, dynamic> json) => AnimeDto(
|
||||
id: (json['mal_id'] as num?)?.toInt(),
|
||||
title: json['title'] as String?,
|
||||
rating: json['rating'] as String?,
|
||||
synopsis: json['synopsis'] as String?,
|
||||
|
@ -6,15 +6,18 @@ import '../dtos/animes_dto.dart';
|
||||
extension AnimesMapper on AnimesDto {
|
||||
HomeData toDomain() => HomeData(
|
||||
data: data?.map((dto) => dto.toDomain()).toList(),
|
||||
nextPage: (pagination?.hasNextPage ?? false) ? ((pagination?.currentPage ?? 0) + 1) : null);
|
||||
nextPage: (pagination?.hasNextPage ?? false)
|
||||
? ((pagination?.currentPage ?? 0) + 1)
|
||||
: null);
|
||||
}
|
||||
|
||||
extension AnimeMapper on AnimeDto {
|
||||
CardData toDomain() => CardData(
|
||||
id: id.toString(),
|
||||
name: title ?? "",
|
||||
imageUrl: images?.jpg?.imageUrl ?? "placeholder.co/250",
|
||||
descr:
|
||||
"Rating: ${rating ?? "unknown"}\nYear: ${year ?? "unknown"}\nType: ${type ?? "unknown"}.\n\n${synopsis ?? "No description provided"} ",
|
||||
cuttedDescr:
|
||||
"Rating: ${rating ?? "unknown"}\nYear: ${year ?? "unknown"}\nType: ${type ?? "unknown"}");
|
||||
type: type,
|
||||
year: year,
|
||||
descr: synopsis,
|
||||
rating: rating);
|
||||
}
|
||||
|
@ -1,10 +1,10 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:flutter_project/components/utils/error_callback.dart';
|
||||
import 'package:flutter_project/data/mappers/animes_mapper.dart';
|
||||
import 'package:flutter_project/data/repositories/api_interface.dart';
|
||||
|
||||
import '../../components/utils/error_callback.dart';
|
||||
import '../../domain/models/home.dart';
|
||||
import '../dtos/animes_dto.dart';
|
||||
|
||||
@ -15,14 +15,22 @@ class AnimeRepository extends ApiInterface {
|
||||
|
||||
@override
|
||||
Future<HomeData?> loadData(
|
||||
{OnErrorCallback onError, String? q, int page = 1, int pageSize = 25}) async {
|
||||
{OnErrorCallback onError,
|
||||
String? q,
|
||||
int page = 1,
|
||||
int pageSize = 25}) async {
|
||||
try {
|
||||
const String url = "$_baseUrl/v4/anime?sfw";
|
||||
|
||||
final Response<dynamic> response = await _dio.get<Map<dynamic, dynamic>>(url,
|
||||
queryParameters: {'q': q, 'page': page, 'limit': !(pageSize > 25) ? pageSize : 25});
|
||||
final Response<dynamic> response = await _dio
|
||||
.get<Map<dynamic, dynamic>>(url, queryParameters: {
|
||||
'q': q,
|
||||
'page': page,
|
||||
'limit': !(pageSize > 25) ? pageSize : 25
|
||||
});
|
||||
|
||||
final AnimesDto dto = AnimesDto.fromJson(response.data as Map<String, dynamic>);
|
||||
final AnimesDto dto =
|
||||
AnimesDto.fromJson(response.data as Map<String, dynamic>);
|
||||
|
||||
final HomeData data = dto.toDomain();
|
||||
|
||||
|
@ -10,23 +10,26 @@ class MockRepository extends ApiInterface {
|
||||
return HomeData(
|
||||
data: [
|
||||
CardData(
|
||||
name: "Test",
|
||||
imageUrl: "https://loremflickr.com/250/150/cat",
|
||||
descr: "Description",
|
||||
cuttedDescr: "cutted",
|
||||
),
|
||||
name: "Test",
|
||||
imageUrl: "https://loremflickr.com/250/150/cat",
|
||||
descr: "descr",
|
||||
type: "Type",
|
||||
year: 2024,
|
||||
rating: "R"),
|
||||
CardData(
|
||||
name: "Test 2",
|
||||
imageUrl: "https://loremflickr.com/200/250/cat",
|
||||
descr: "Description",
|
||||
cuttedDescr: "cutted",
|
||||
),
|
||||
name: "Test 2",
|
||||
imageUrl: "https://loremflickr.com/200/250/cat",
|
||||
descr: "descr",
|
||||
type: "Type",
|
||||
year: 2024,
|
||||
rating: "R"),
|
||||
CardData(
|
||||
name: "Test 3",
|
||||
imageUrl: "https://loremflickr.com/200/200/cat",
|
||||
descr: "Description",
|
||||
cuttedDescr: "cutted",
|
||||
),
|
||||
name: "Test 3",
|
||||
imageUrl: "https://loremflickr.com/200/200/cat",
|
||||
descr: "descr",
|
||||
type: "Type",
|
||||
year: 2024,
|
||||
rating: "R"),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
@ -1,9 +1,18 @@
|
||||
class CardData {
|
||||
final String name;
|
||||
final String imageUrl;
|
||||
final String descr;
|
||||
final String cuttedDescr;
|
||||
final String? type;
|
||||
final int? year;
|
||||
final String? rating;
|
||||
final String? descr;
|
||||
final String? id;
|
||||
|
||||
const CardData(
|
||||
{required this.name, required this.imageUrl, required this.descr, required this.cuttedDescr});
|
||||
{required this.name,
|
||||
required this.imageUrl,
|
||||
required this.type,
|
||||
required this.year,
|
||||
required this.rating,
|
||||
required this.descr,
|
||||
this.id});
|
||||
}
|
||||
|
@ -1,9 +1,14 @@
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||
import 'package:flutter_project/components/locale/l10n/app_localizations.dart';
|
||||
import 'package:flutter_project/data/repositories/anime_repository.dart';
|
||||
import 'package:flutter_project/views/home_page/bloc/bloc.dart';
|
||||
import 'package:flutter_project/views/home_page/home_bloc/bloc.dart';
|
||||
import 'package:flutter_project/views/home_page/home_page.dart';
|
||||
import 'package:flutter_project/views/like_bloc/like_bloc.dart';
|
||||
import 'package:flutter_project/views/locale_bloc/locale_bloc.dart';
|
||||
import 'package:flutter_project/views/locale_bloc/locale_state.dart';
|
||||
|
||||
void main() {
|
||||
runApp(const MyApp());
|
||||
@ -14,21 +19,31 @@ class MyApp extends StatelessWidget {
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return RepositoryProvider<AnimeRepository>(
|
||||
lazy: true,
|
||||
create: (_) => AnimeRepository(),
|
||||
child: BlocProvider<HomeBloc>(
|
||||
return BlocProvider<LikeBloc>(
|
||||
lazy: false,
|
||||
create: (context) => LikeBloc(),
|
||||
child: BlocProvider<LocaleBloc>(
|
||||
lazy: false,
|
||||
create: (context) => HomeBloc(context.read<AnimeRepository>()),
|
||||
child: MaterialApp(
|
||||
title: 'Anime list',
|
||||
localizationsDelegates: AppLocale.localizationsDelegates,
|
||||
supportedLocales: AppLocale.supportedLocales,
|
||||
theme: ThemeData(
|
||||
colorScheme: ColorScheme.fromSeed(seedColor: Colors.lightGreen),
|
||||
useMaterial3: true,
|
||||
create: (context) => LocaleBloc(Locale(Platform.localeName)),
|
||||
child: BlocBuilder<LocaleBloc, LocaleState>(
|
||||
builder: (context, state) => MaterialApp(
|
||||
title: 'Anime list',
|
||||
locale: state.currentLocale,
|
||||
localizationsDelegates: AppLocale.localizationsDelegates,
|
||||
supportedLocales: AppLocale.supportedLocales,
|
||||
theme: ThemeData(
|
||||
colorScheme: ColorScheme.fromSeed(seedColor: Colors.lightGreen),
|
||||
useMaterial3: true,
|
||||
),
|
||||
home: RepositoryProvider<AnimeRepository>(
|
||||
lazy: true,
|
||||
create: (_) => AnimeRepository(),
|
||||
child: BlocProvider<HomeBloc>(
|
||||
lazy: false,
|
||||
create: (context) =>
|
||||
HomeBloc(context.read<AnimeRepository>()),
|
||||
child: const HomePage())),
|
||||
),
|
||||
home: const HomePage(),
|
||||
),
|
||||
),
|
||||
);
|
||||
|
@ -1,11 +1,13 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_project/components/locale/l10n/app_localizations.dart';
|
||||
|
||||
import '../../domain/models/card.dart';
|
||||
|
||||
class DetailsPage extends StatelessWidget {
|
||||
final AppLocale locale;
|
||||
final CardData data;
|
||||
|
||||
const DetailsPage(this.data, {super.key});
|
||||
const DetailsPage(this.locale, this.data, {super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
@ -38,7 +40,7 @@ class DetailsPage extends StatelessWidget {
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(10),
|
||||
child: Text(
|
||||
data.descr,
|
||||
'${locale.apiYear}: ${data.year ?? locale.unknown}, ${locale.apiType}: ${data.type ?? locale.unknown}, ${locale.apiRating}: ${data.rating ?? locale.unknown}\n${locale.apiDesc != "" ? "\n${locale.apiDesc}\n" : ""}\n${data.descr ?? locale.apiNoDesc}',
|
||||
style: Theme.of(context).textTheme.bodyLarge,
|
||||
),
|
||||
)
|
||||
|
@ -1,36 +1,49 @@
|
||||
part of 'home_page.dart';
|
||||
|
||||
typedef onLikeCallback = void Function(String title, bool isLiked)?;
|
||||
typedef onLikeCallback = void Function(String? id, String title, bool isLiked)?;
|
||||
|
||||
class _Card extends StatefulWidget {
|
||||
class _Card extends StatelessWidget {
|
||||
final AppLocale locale;
|
||||
final String name;
|
||||
final String imageUrl;
|
||||
final String descr;
|
||||
final String? type;
|
||||
final int? year;
|
||||
final String? rating;
|
||||
final String? id;
|
||||
final onLikeCallback onLike;
|
||||
final VoidCallback? onTap;
|
||||
final bool isLiked;
|
||||
|
||||
const _Card(
|
||||
{required this.name, required this.imageUrl, this.onLike, this.onTap, required this.descr});
|
||||
{required this.locale,
|
||||
required this.name,
|
||||
required this.imageUrl,
|
||||
required this.type,
|
||||
required this.year,
|
||||
required this.rating,
|
||||
this.id,
|
||||
this.onLike,
|
||||
this.onTap,
|
||||
this.isLiked = false});
|
||||
|
||||
factory _Card.withData(CardData d, {onLikeCallback onLike, VoidCallback? onTap}) => _Card(
|
||||
name: d.name,
|
||||
imageUrl: d.imageUrl,
|
||||
onLike: onLike,
|
||||
onTap: onTap,
|
||||
descr: d.cuttedDescr,
|
||||
);
|
||||
|
||||
@override
|
||||
State<_Card> createState() => _CardState();
|
||||
}
|
||||
|
||||
class _CardState extends State<_Card> {
|
||||
bool isLiked = false;
|
||||
factory _Card.withData(AppLocale locale, CardData d,
|
||||
{onLikeCallback onLike, VoidCallback? onTap, bool isLiked = false}) =>
|
||||
_Card(
|
||||
locale: locale,
|
||||
name: d.name,
|
||||
imageUrl: d.imageUrl,
|
||||
onLike: onLike,
|
||||
onTap: onTap,
|
||||
type: d.type,
|
||||
year: d.year,
|
||||
rating: d.rating,
|
||||
isLiked: isLiked,
|
||||
id: d.id);
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return GestureDetector(
|
||||
onTap: widget.onTap,
|
||||
onTap: onTap,
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Container(
|
||||
@ -51,14 +64,15 @@ class _CardState extends State<_Card> {
|
||||
children: [
|
||||
ClipRRect(
|
||||
borderRadius: BorderRadius.only(
|
||||
topLeft: Radius.circular(20), bottomLeft: Radius.circular(20)),
|
||||
topLeft: Radius.circular(20),
|
||||
bottomLeft: Radius.circular(20)),
|
||||
child: ConstrainedBox(
|
||||
constraints: BoxConstraints(minHeight: 200),
|
||||
child: SizedBox(
|
||||
width: 150,
|
||||
height: double.infinity,
|
||||
child: Image.network(
|
||||
widget.imageUrl,
|
||||
imageUrl,
|
||||
fit: BoxFit.cover,
|
||||
),
|
||||
),
|
||||
@ -71,11 +85,11 @@ class _CardState extends State<_Card> {
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
widget.name,
|
||||
name,
|
||||
style: Theme.of(context).textTheme.headlineSmall,
|
||||
),
|
||||
Text(
|
||||
widget.descr,
|
||||
'${locale.apiYear}: ${year ?? locale.unknown}, ${locale.apiType}: ${type ?? locale.unknown}, ${locale.apiRating}: ${rating ?? locale.unknown}',
|
||||
style: Theme.of(context).textTheme.labelSmall,
|
||||
),
|
||||
],
|
||||
@ -87,12 +101,7 @@ class _CardState extends State<_Card> {
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(top: 16.0, right: 8.0),
|
||||
child: GestureDetector(
|
||||
onTap: () {
|
||||
setState(() {
|
||||
isLiked = !isLiked;
|
||||
});
|
||||
widget.onLike?.call(widget.name, isLiked);
|
||||
},
|
||||
onTap: () => onLike?.call(id, name, isLiked),
|
||||
child: AnimatedSwitcher(
|
||||
duration: const Duration(milliseconds: 150),
|
||||
child: isLiked
|
||||
|
@ -1,7 +1,7 @@
|
||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||
import 'package:flutter_project/data/repositories/anime_repository.dart';
|
||||
import 'package:flutter_project/views/home_page/bloc/events.dart';
|
||||
import 'package:flutter_project/views/home_page/bloc/state.dart';
|
||||
import 'package:flutter_project/views/home_page/home_bloc/events.dart';
|
||||
import 'package:flutter_project/views/home_page/home_bloc/state.dart';
|
||||
|
||||
class HomeBloc extends Bloc<HomeEvent, HomeState> {
|
||||
final AnimeRepository repo;
|
||||
@ -11,7 +11,8 @@ class HomeBloc extends Bloc<HomeEvent, HomeState> {
|
||||
on<HomeShowButtonToTopEvent>(_onShowButton);
|
||||
}
|
||||
|
||||
void _onShowButton(HomeShowButtonToTopEvent event, Emitter<HomeState> emit) {
|
||||
Future<void> _onShowButton(
|
||||
HomeShowButtonToTopEvent event, Emitter<HomeState> emit) async {
|
||||
if (event.isShown != null) {
|
||||
if (event.isShown == true) {
|
||||
emit(state.copyWith(isButtonToTopShown: true));
|
||||
@ -21,7 +22,8 @@ class HomeBloc extends Bloc<HomeEvent, HomeState> {
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _onLoadData(HomeLoadDataEvent event, Emitter<HomeState> emit) async {
|
||||
Future<void> _onLoadData(
|
||||
HomeLoadDataEvent event, Emitter<HomeState> emit) async {
|
||||
if (event.hasError != null && event.hasError == true) {
|
||||
emit(state.copyWith(error: null));
|
||||
}
|
||||
@ -34,13 +36,17 @@ class HomeBloc extends Bloc<HomeEvent, HomeState> {
|
||||
|
||||
String? error;
|
||||
|
||||
final data =
|
||||
await repo.loadData(q: event.search, page: event.nextPage ?? 1, onError: (e) => error = e);
|
||||
final data = await repo.loadData(
|
||||
q: event.search, page: event.nextPage ?? 1, onError: (e) => error = e);
|
||||
|
||||
if (event.nextPage != null) {
|
||||
data?.data?.insertAll(0, state.data?.data ?? []);
|
||||
}
|
||||
|
||||
emit(state.copyWith(data: data, isLoading: false, isPaginationLoading: false, error: error));
|
||||
emit(state.copyWith(
|
||||
data: data,
|
||||
isLoading: false,
|
||||
isPaginationLoading: false,
|
||||
error: error));
|
||||
}
|
||||
}
|
@ -2,14 +2,21 @@ import 'package:flutter/cupertino.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||
import 'package:flutter_project/components/extensions/context_x.dart';
|
||||
import 'package:flutter_project/components/locale/l10n/app_localizations.dart';
|
||||
import 'package:flutter_project/components/utils/debounce.dart';
|
||||
import 'package:flutter_project/domain/models/card.dart';
|
||||
import 'package:flutter_project/views/common/svg_objects.dart';
|
||||
import 'package:flutter_project/views/details_page/details_page.dart';
|
||||
import 'package:flutter_project/views/home_page/bloc/events.dart';
|
||||
import 'package:flutter_project/views/home_page/bloc/state.dart';
|
||||
import 'package:flutter_project/views/home_page/home_bloc/events.dart';
|
||||
import 'package:flutter_project/views/home_page/home_bloc/state.dart';
|
||||
import 'package:flutter_project/views/like_bloc/like_events.dart';
|
||||
import 'package:flutter_project/views/like_bloc/like_state.dart';
|
||||
import 'package:flutter_project/views/locale_bloc/locale_bloc.dart';
|
||||
import 'package:flutter_project/views/locale_bloc/locale_events.dart';
|
||||
import 'package:flutter_project/views/locale_bloc/locale_state.dart';
|
||||
|
||||
import 'bloc/bloc.dart';
|
||||
import '../like_bloc/like_bloc.dart';
|
||||
import 'home_bloc/bloc.dart';
|
||||
|
||||
part 'card.dart';
|
||||
|
||||
@ -29,7 +36,7 @@ class _HomePageState extends State<HomePage> {
|
||||
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
|
||||
title: Center(
|
||||
child: Text(
|
||||
'Anime list',
|
||||
context.locale.appBarTitle,
|
||||
),
|
||||
),
|
||||
),
|
||||
@ -53,8 +60,10 @@ class _BodyState extends State<Body> {
|
||||
void initState() {
|
||||
SvgObjects.init();
|
||||
|
||||
WidgetsBinding.instance.addPostFrameCallback(
|
||||
(_) => context.read<HomeBloc>().add(const HomeLoadDataEvent()));
|
||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||
context.read<HomeBloc>().add(const HomeLoadDataEvent());
|
||||
context.read<LikeBloc>().add(const LoadLikesEvent());
|
||||
});
|
||||
scrollController.addListener(_viewListScrollListener);
|
||||
|
||||
super.initState();
|
||||
@ -72,19 +81,40 @@ class _BodyState extends State<Body> {
|
||||
return Stack(children: [
|
||||
Column(
|
||||
children: [
|
||||
Padding(
|
||||
padding: EdgeInsets.only(left: 16, right: 16, top: 10, bottom: 10),
|
||||
child: Center(
|
||||
child: CupertinoSearchTextField(
|
||||
placeholder: context.locale.search,
|
||||
controller: searchController,
|
||||
onChanged: (search) {
|
||||
Debounce.run(() => context
|
||||
.read<HomeBloc>()
|
||||
.add(HomeLoadDataEvent(search: search)));
|
||||
},
|
||||
Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: Padding(
|
||||
padding: EdgeInsets.only(left: 16, top: 10, bottom: 10),
|
||||
child: Center(
|
||||
child: CupertinoSearchTextField(
|
||||
placeholder: context.locale.search,
|
||||
controller: searchController,
|
||||
onChanged: (search) {
|
||||
Debounce.run(() => context
|
||||
.read<HomeBloc>()
|
||||
.add(HomeLoadDataEvent(search: search)));
|
||||
},
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
GestureDetector(
|
||||
onTap: () =>
|
||||
context.read<LocaleBloc>().add(const ChangeLocaleEvent()),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
child: BlocBuilder<LocaleBloc, LocaleState>(
|
||||
builder: (context, state) {
|
||||
return AnimatedSwitcher(
|
||||
duration: const Duration(milliseconds: 250),
|
||||
child: state.currentLocale.languageCode == 'ru'
|
||||
? const SvgRu(key: ValueKey<int>(1))
|
||||
: const SvgUs(key: ValueKey<int>(0)));
|
||||
}),
|
||||
),
|
||||
)
|
||||
],
|
||||
),
|
||||
BlocConsumer<HomeBloc, HomeState>(
|
||||
listener: (context, state) {
|
||||
@ -94,25 +124,29 @@ class _BodyState extends State<Body> {
|
||||
},
|
||||
builder: (context, state) => state.isLoading
|
||||
? const CircularProgressIndicator()
|
||||
: Expanded(
|
||||
child: RefreshIndicator(
|
||||
onRefresh: _onRefresh,
|
||||
child: ListView.builder(
|
||||
controller: scrollController,
|
||||
padding: EdgeInsets.only(bottom: 10),
|
||||
itemCount: state.data?.data?.length ?? 0,
|
||||
itemBuilder: (context, index) {
|
||||
final data = state.data?.data?[index];
|
||||
: BlocBuilder<LikeBloc, LikeState>(
|
||||
builder: (context, likeState) => Expanded(
|
||||
child: RefreshIndicator(
|
||||
onRefresh: _onRefresh,
|
||||
child: ListView.builder(
|
||||
controller: scrollController,
|
||||
padding: EdgeInsets.only(bottom: 10),
|
||||
itemCount: state.data?.data?.length ?? 0,
|
||||
itemBuilder: (context, index) {
|
||||
final data = state.data?.data?[index];
|
||||
|
||||
return data != null
|
||||
? _Card.withData(data,
|
||||
onLike: (title, isLiked) =>
|
||||
_showSnackBar(context, isLiked, title),
|
||||
onTap: () => _navToDetails(context, data))
|
||||
: const SizedBox.shrink();
|
||||
},
|
||||
),
|
||||
)),
|
||||
return data != null
|
||||
? _Card.withData(context.locale, data,
|
||||
isLiked:
|
||||
likeState.likedIds?.contains(data.id) ==
|
||||
true,
|
||||
onLike: _onLike,
|
||||
onTap: () => _navToDetails(context, data))
|
||||
: const SizedBox.shrink();
|
||||
},
|
||||
),
|
||||
)),
|
||||
),
|
||||
),
|
||||
BlocBuilder<HomeBloc, HomeState>(
|
||||
builder: (context, state) => state.isPaginationLoading
|
||||
@ -134,6 +168,13 @@ class _BodyState extends State<Body> {
|
||||
]);
|
||||
}
|
||||
|
||||
void _onLike(String? id, String title, bool isLiked) {
|
||||
if (id != null) {
|
||||
context.read<LikeBloc>().add(ChangeLikeEvent(id));
|
||||
_showSnackBar(context, !isLiked, title);
|
||||
}
|
||||
}
|
||||
|
||||
void _onErrorShowDialog(BuildContext context, HomeState state) {
|
||||
showCupertinoDialog(
|
||||
context: context,
|
||||
@ -207,6 +248,8 @@ class _BodyState extends State<Body> {
|
||||
|
||||
void _navToDetails(BuildContext context, CardData d) {
|
||||
Navigator.push(
|
||||
context, CupertinoPageRoute(builder: (context) => DetailsPage(d)));
|
||||
context,
|
||||
CupertinoPageRoute(
|
||||
builder: (context) => DetailsPage(context.locale, d)));
|
||||
}
|
||||
}
|
||||
|
37
lib/views/like_bloc/like_bloc.dart
Normal file
37
lib/views/like_bloc/like_bloc.dart
Normal file
@ -0,0 +1,37 @@
|
||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||
import 'package:flutter_project/views/like_bloc/like_events.dart';
|
||||
import 'package:flutter_project/views/like_bloc/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));
|
||||
}
|
||||
}
|
12
lib/views/like_bloc/like_events.dart
Normal file
12
lib/views/like_bloc/like_events.dart
Normal file
@ -0,0 +1,12 @@
|
||||
abstract class LikeEvent {
|
||||
const LikeEvent();
|
||||
}
|
||||
|
||||
class LoadLikesEvent extends LikeEvent {
|
||||
const LoadLikesEvent();
|
||||
}
|
||||
|
||||
class ChangeLikeEvent extends LikeEvent {
|
||||
final String id;
|
||||
const ChangeLikeEvent(this.id);
|
||||
}
|
14
lib/views/like_bloc/like_state.dart
Normal file
14
lib/views/like_bloc/like_state.dart
Normal file
@ -0,0 +1,14 @@
|
||||
import 'package:copy_with_extension/copy_with_extension.dart';
|
||||
import 'package:equatable/equatable.dart';
|
||||
|
||||
part 'like_state.g.dart';
|
||||
|
||||
@CopyWith()
|
||||
class LikeState extends Equatable {
|
||||
final List<String>? likedIds;
|
||||
|
||||
const LikeState({this.likedIds});
|
||||
|
||||
@override
|
||||
List<Object?> get props => [likedIds];
|
||||
}
|
56
lib/views/like_bloc/like_state.g.dart
Normal file
56
lib/views/like_bloc/like_state.g.dart
Normal file
@ -0,0 +1,56 @@
|
||||
// GENERATED CODE - DO NOT MODIFY BY HAND
|
||||
|
||||
part of 'like_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);
|
||||
}
|
21
lib/views/locale_bloc/locale_bloc.dart
Normal file
21
lib/views/locale_bloc/locale_bloc.dart
Normal file
@ -0,0 +1,21 @@
|
||||
import 'dart:ui';
|
||||
|
||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||
import 'package:flutter_project/components/locale/l10n/app_localizations.dart';
|
||||
|
||||
import 'locale_events.dart';
|
||||
import '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(
|
||||
(loc) => loc.languageCode != state.currentLocale.languageCode);
|
||||
emit(state.copyWith(currentLocale: toChange));
|
||||
}
|
||||
}
|
7
lib/views/locale_bloc/locale_events.dart
Normal file
7
lib/views/locale_bloc/locale_events.dart
Normal file
@ -0,0 +1,7 @@
|
||||
abstract class LocaleEvent {
|
||||
const LocaleEvent();
|
||||
}
|
||||
|
||||
class ChangeLocaleEvent extends LocaleEvent {
|
||||
const ChangeLocaleEvent();
|
||||
}
|
16
lib/views/locale_bloc/locale_state.dart
Normal file
16
lib/views/locale_bloc/locale_state.dart
Normal file
@ -0,0 +1,16 @@
|
||||
import 'dart:ui';
|
||||
|
||||
import 'package:copy_with_extension/copy_with_extension.dart';
|
||||
import 'package:equatable/equatable.dart';
|
||||
|
||||
part 'locale_state.g.dart';
|
||||
|
||||
@CopyWith()
|
||||
class LocaleState extends Equatable {
|
||||
final Locale currentLocale;
|
||||
|
||||
const LocaleState({required this.currentLocale});
|
||||
|
||||
@override
|
||||
List<Object?> get props => [currentLocale];
|
||||
}
|
58
lib/views/locale_bloc/locale_state.g.dart
Normal file
58
lib/views/locale_bloc/locale_state.g.dart
Normal file
@ -0,0 +1,58 @@
|
||||
// GENERATED CODE - DO NOT MODIFY BY HAND
|
||||
|
||||
part of 'locale_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() || currentLocale == null
|
||||
? _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);
|
||||
}
|
@ -183,7 +183,7 @@ packages:
|
||||
source: hosted
|
||||
version: "3.1.1"
|
||||
copy_with_extension:
|
||||
dependency: "direct dev"
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: copy_with_extension
|
||||
sha256: fbcf890b0c34aedf0894f91a11a579994b61b4e04080204656b582708b5b1125
|
||||
@ -191,7 +191,7 @@ packages:
|
||||
source: hosted
|
||||
version: "5.0.4"
|
||||
copy_with_extension_gen:
|
||||
dependency: "direct dev"
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: copy_with_extension_gen
|
||||
sha256: "51cd11094096d40824c8da629ca7f16f3b7cea5fc44132b679617483d43346b0"
|
||||
@ -627,7 +627,7 @@ packages:
|
||||
source: hosted
|
||||
version: "1.3.0"
|
||||
shared_preferences:
|
||||
dependency: "direct dev"
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: shared_preferences
|
||||
sha256: "746e5369a43170c25816cc472ee016d3a66bc13fcf430c0bc41ad7b4b2922051"
|
||||
|
@ -41,6 +41,9 @@ dependencies:
|
||||
pretty_dio_logger: ^1.4.0
|
||||
flutter_bloc: ^8.1.6
|
||||
equatable: ^2.0.5
|
||||
copy_with_extension: ^5.0.4
|
||||
copy_with_extension_gen: ^5.0.4
|
||||
shared_preferences: ^2.3.2
|
||||
|
||||
dev_dependencies:
|
||||
flutter_test:
|
||||
@ -60,10 +63,6 @@ dev_dependencies:
|
||||
sdk: flutter
|
||||
intl: ^0.19.0
|
||||
|
||||
|
||||
shared_preferences: ^2.3.2
|
||||
copy_with_extension_gen: ^5.0.4
|
||||
copy_with_extension: ^5.0.4
|
||||
json_serializable: ^6.7.1
|
||||
build_runner: ^2.4.9
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user