added separate setting to change currency

This commit is contained in:
ShabOl 2024-12-30 11:57:56 +04:00
parent f0a17af845
commit 1377a5dbdd
22 changed files with 495 additions and 176 deletions

View File

@ -10,9 +10,12 @@
"addedToFavourite": "is added to favourites", "addedToFavourite": "is added to favourites",
"removedFromFavourite": "is removed from favourites", "removedFromFavourite": "is removed from favourites",
"cardsLoadingFailed": "Server is unreachable",
"cardsLoadingFailedSnackBar": "Failed to load crypto data",
"coinDataPriceChange": "for the last 24 hours", "coinDataPriceChange": "for the last 24 hours",
"settingsLanguage": "Language", "settingsLanguage": "Language",
"settingsCurrency": "Currency",
"navigationHome": "Home", "navigationHome": "Home",
"navigationFavourites": "Favourites", "navigationFavourites": "Favourites",

View File

@ -10,9 +10,12 @@
"addedToFavourite": "добавлен в избранное", "addedToFavourite": "добавлен в избранное",
"removedFromFavourite": "удален из избранного", "removedFromFavourite": "удален из избранного",
"cardsLoadingFailed": "Сервер недоступен",
"cardsLoadingFailedSnackBar": "Не удалось загрузить данные о криптовалюте",
"coinDataPriceChange": "за последние 24 часа", "coinDataPriceChange": "за последние 24 часа",
"settingsLanguage": "Язык", "settingsLanguage": "Язык",
"settingsCurrency": "Валюта",
"navigationHome": "Главная", "navigationHome": "Главная",
"navigationFavourites": "Избранное", "navigationFavourites": "Избранное",

View File

@ -137,6 +137,18 @@ abstract class AppLocale {
/// **'is removed from favourites'** /// **'is removed from favourites'**
String get removedFromFavourite; String get removedFromFavourite;
/// No description provided for @cardsLoadingFailed.
///
/// In en, this message translates to:
/// **'Server is unreachable'**
String get cardsLoadingFailed;
/// No description provided for @cardsLoadingFailedSnackBar.
///
/// In en, this message translates to:
/// **'Failed to load crypto data'**
String get cardsLoadingFailedSnackBar;
/// No description provided for @coinDataPriceChange. /// No description provided for @coinDataPriceChange.
/// ///
/// In en, this message translates to: /// In en, this message translates to:
@ -149,6 +161,12 @@ abstract class AppLocale {
/// **'Language'** /// **'Language'**
String get settingsLanguage; String get settingsLanguage;
/// No description provided for @settingsCurrency.
///
/// In en, this message translates to:
/// **'Currency'**
String get settingsCurrency;
/// No description provided for @navigationHome. /// No description provided for @navigationHome.
/// ///
/// In en, this message translates to: /// In en, this message translates to:

View File

@ -27,12 +27,21 @@ class AppLocaleEn extends AppLocale {
@override @override
String get removedFromFavourite => 'is removed from favourites'; String get removedFromFavourite => 'is removed from favourites';
@override
String get cardsLoadingFailed => 'Server is unreachable';
@override
String get cardsLoadingFailedSnackBar => 'Failed to load crypto data';
@override @override
String get coinDataPriceChange => 'for the last 24 hours'; String get coinDataPriceChange => 'for the last 24 hours';
@override @override
String get settingsLanguage => 'Language'; String get settingsLanguage => 'Language';
@override
String get settingsCurrency => 'Currency';
@override @override
String get navigationHome => 'Home'; String get navigationHome => 'Home';

View File

@ -27,12 +27,21 @@ class AppLocaleRu extends AppLocale {
@override @override
String get removedFromFavourite => 'удален из избранного'; String get removedFromFavourite => 'удален из избранного';
@override
String get cardsLoadingFailed => 'Сервер недоступен';
@override
String get cardsLoadingFailedSnackBar => 'Не удалось загрузить данные о криптовалюте';
@override @override
String get coinDataPriceChange => 'за последние 24 часа'; String get coinDataPriceChange => 'за последние 24 часа';
@override @override
String get settingsLanguage => 'Язык'; String get settingsLanguage => 'Язык';
@override
String get settingsCurrency => 'Валюта';
@override @override
String get navigationHome => 'Главная'; String get navigationHome => 'Главная';

View File

@ -4,28 +4,26 @@ import 'package:flutter_android_app/domain/models/card.dart';
import 'package:flutter_android_app/domain/models/home.dart'; import 'package:flutter_android_app/domain/models/home.dart';
extension CoinDataDtoToModel on CoinDataDto { extension CoinDataDtoToModel on CoinDataDto {
CardData toDomain(AppLocale? locale) => CardData( CardData toDomain(AppLocale? locale, String currencyId) => CardData(
id: id ?? 'UNKNOWN', id: id ?? 'UNKNOWN',
title: name ?? 'UNKNOWN', title: name ?? 'UNKNOWN',
imageUrl: image, imageUrl: image,
currentPrice: _getLocalizedPrice(currentPrice, locale?.localeName), currentPrice: '$currentPrice ${_getCurrencySymbol(currencyId)}',
priceChange: _getLocalizedPriceChange(priceChange24h, locale), priceChange: _getLocalizedPriceChange(priceChange24h, locale, currencyId),
); );
String _getLocalizedPrice(double? price, String? localeName) { String _getCurrencySymbol(String currencyId) {
if (localeName == null) { return switch (currencyId) {
return '$price \$'; 'rub' => '',
} 'usd' => '\$',
return switch (localeName) { _ => '?',
'ru' => '$price',
_ => '$price \$',
}; };
} }
String _getLocalizedPriceChange(double? priceChange, AppLocale? locale) { String _getLocalizedPriceChange(double? priceChange, AppLocale? locale, String currencyId) {
if (priceChange == null) { if (priceChange == null) {
return '+${_getLocalizedPrice(0, locale?.localeName)}'; return '+0 ${_getCurrencySymbol(currencyId)}';
} }
String retVal = ''; String retVal = '';
@ -33,15 +31,15 @@ extension CoinDataDtoToModel on CoinDataDto {
retVal += '+'; retVal += '+';
} }
retVal += _getLocalizedPrice(priceChange, locale?.localeName); retVal += '$priceChange ${_getCurrencySymbol(currencyId)}';
return '$retVal ${locale?.coinDataPriceChange}'; return '$retVal ${locale?.coinDataPriceChange}';
} }
} }
extension CoinsDtoToModel on CoinsDto { extension CoinsDtoToModel on CoinsDto {
HomeData toDomain(AppLocale? locale, int currentPage) => HomeData( HomeData toDomain(AppLocale? locale, String currencyId, int currentPage) => HomeData(
data: coins?.map((e) => e.toDomain(locale)).toList(), data: coins?.map((e) => e.toDomain(locale, currencyId)).toList(),
nextPage: currentPage + 1, nextPage: currentPage + 1,
); );
} }

View File

@ -1,6 +1,7 @@
import 'dart:io'; import 'dart:io';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_android_app/presentation/currency_bloc/currency_bloc.dart';
import 'package:flutter_android_app/presentation/favourites_bloc/favourites_bloc.dart'; import 'package:flutter_android_app/presentation/favourites_bloc/favourites_bloc.dart';
import 'package:flutter_android_app/presentation/home_page/bloc/bloc.dart'; import 'package:flutter_android_app/presentation/home_page/bloc/bloc.dart';
import 'package:flutter_android_app/presentation/home_page/home_page.dart'; import 'package:flutter_android_app/presentation/home_page/home_page.dart';
@ -30,33 +31,37 @@ class _MyAppState extends State<MyApp> {
return BlocProvider<FavouritesBloc>( return BlocProvider<FavouritesBloc>(
lazy: false, lazy: false,
create: (context) => FavouritesBloc(), create: (context) => FavouritesBloc(),
child: BlocProvider<LocaleBloc>( child: BlocProvider<CurrencyBloc>(
lazy: false, lazy: false,
create: (context) => LocaleBloc(Locale(_getLangCode(Platform.localeName))), create: (context) => CurrencyBloc(),
child: BlocBuilder<LocaleBloc, LocaleState>( child: BlocProvider<LocaleBloc>(
builder: (context, state) => MaterialApp( lazy: false,
title: 'Cryptocurrency Exchange App', create: (context) => LocaleBloc(Locale(_getLangCode(Platform.localeName))),
locale: state.currentLocale, child: BlocBuilder<LocaleBloc, LocaleState>(
localizationsDelegates: AppLocale.localizationsDelegates, builder: (context, state) => MaterialApp(
supportedLocales: AppLocale.supportedLocales, title: 'Cryptocurrency Exchange App',
theme: ThemeData( locale: state.currentLocale,
colorScheme: ColorScheme.fromSeed( localizationsDelegates: AppLocale.localizationsDelegates,
seedColor: Colors.indigoAccent, supportedLocales: AppLocale.supportedLocales,
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(
seedColor: Colors.indigoAccent,
brightness: isDarkMode ? Brightness.dark : Brightness.light,
),
useMaterial3: true,
brightness: isDarkMode ? Brightness.dark : Brightness.light, brightness: isDarkMode ? Brightness.dark : Brightness.light,
), ),
useMaterial3: true, debugShowCheckedModeBanner: false,
brightness: isDarkMode ? Brightness.dark : Brightness.light, home: RepositoryProvider<CryptoRepository>(
), lazy: true,
debugShowCheckedModeBanner: false, create: (_) => CryptoRepository(),
home: RepositoryProvider<CryptoRepository>( child: BlocProvider<HomeBloc>(
lazy: true, lazy: false,
create: (_) => CryptoRepository(), create: (context) => HomeBloc(context.read<CryptoRepository>()),
child: BlocProvider<HomeBloc>( child: MainScaffold(
lazy: false, toggleDarkMode: _toggleDarkMode,
create: (context) => HomeBloc(context.read<CryptoRepository>()), isDarkModeSelected: isDarkMode,
child: MainScaffold( ),
toggleDarkMode: _toggleDarkMode,
isDarkModeSelected: isDarkMode,
), ),
), ),
), ),

View File

@ -0,0 +1,61 @@
import 'package:flutter_android_app/presentation/currency_bloc/currency_events.dart';
import 'package:flutter_android_app/presentation/currency_bloc/currency_state.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:shared_preferences/shared_preferences.dart';
class CurrencyBloc extends Bloc<CurrencyEvent, CurrencyState> {
static const String _currencyPrefsKey = 'local_currency';
static const List<String> _availableCurrencyIds = ['usd', 'rub'];
static final String _defaultCurrencyId = _availableCurrencyIds[0];
CurrencyBloc() : super(const CurrencyState()) {
on<LoadLocalCurrencyEvent>(_onLoadLocalCurrency);
on<ToggleLocalCurrencyEvent>(_onToggleLocalCurrency);
}
Future<void> _onLoadLocalCurrency(
LoadLocalCurrencyEvent event, Emitter<CurrencyState> emit
) async {
emit(state.copyWith(
hasCurrencyLoaded: false,
));
final prefs = await SharedPreferences.getInstance();
final data = prefs.getString(_currencyPrefsKey);
String currencyId = '';
if (data != null) {
currencyId = data;
} else {
currencyId = _defaultCurrencyId;
}
emit(state.copyWith(
currencyId: currencyId,
hasCurrencyLoaded: true,
));
}
Future<void> _onToggleLocalCurrency(
ToggleLocalCurrencyEvent event, Emitter<CurrencyState> emit
) async {
if (state.currencyId == null) {
return;
}
final int oldCurrencyIdIdx = _availableCurrencyIds.indexOf(state.currencyId!);
int newCurrencyIdIdx = oldCurrencyIdIdx + 1;
if (newCurrencyIdIdx >= _availableCurrencyIds.length){
newCurrencyIdIdx = 0;
}
final newCurrencyId = _availableCurrencyIds[newCurrencyIdIdx];
final prefs = await SharedPreferences.getInstance();
prefs.setString(_currencyPrefsKey, newCurrencyId);
emit(state.copyWith(
currencyId: newCurrencyId,
));
}
}

View File

@ -0,0 +1,11 @@
abstract class CurrencyEvent {
const CurrencyEvent();
}
class LoadLocalCurrencyEvent extends CurrencyEvent {
const LoadLocalCurrencyEvent();
}
class ToggleLocalCurrencyEvent extends CurrencyEvent {
const ToggleLocalCurrencyEvent();
}

View File

@ -0,0 +1,21 @@
import 'package:copy_with_extension/copy_with_extension.dart';
import 'package:equatable/equatable.dart';
part 'currency_state.g.dart';
@CopyWith()
class CurrencyState extends Equatable {
final String? currencyId;
final bool hasCurrencyLoaded;
const CurrencyState({
this.currencyId,
this.hasCurrencyLoaded = false
});
@override
List<Object?> get props => [
currencyId,
hasCurrencyLoaded,
];
}

View File

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

View File

@ -22,7 +22,7 @@ class FavouritesBloc extends Bloc<FavouritesEvent, FavouritesState> {
final data = prefs.getStringList(_likedPrefsKey); final data = prefs.getStringList(_likedPrefsKey);
emit(state.copyWith( emit(state.copyWith(
likedIds: data, favouritesIds: data,
hasFavouritesLoaded: true, hasFavouritesLoaded: true,
)); ));
} }
@ -42,7 +42,7 @@ class FavouritesBloc extends Bloc<FavouritesEvent, FavouritesState> {
prefs.setStringList(_likedPrefsKey, updatedList); prefs.setStringList(_likedPrefsKey, updatedList);
emit(state.copyWith( emit(state.copyWith(
likedIds: updatedList, favouritesIds: updatedList,
hasFavouritesLoaded: true, hasFavouritesLoaded: true,
)); ));
} }

View File

@ -7,7 +7,7 @@ part of 'favourites_state.dart';
// ************************************************************************** // **************************************************************************
abstract class _$FavouritesStateCWProxy { abstract class _$FavouritesStateCWProxy {
FavouritesState likedIds(List<String>? likedIds); FavouritesState favouritesIds(List<String>? favouritesIds);
FavouritesState hasFavouritesLoaded(bool hasFavouritesLoaded); FavouritesState hasFavouritesLoaded(bool hasFavouritesLoaded);
@ -18,7 +18,7 @@ abstract class _$FavouritesStateCWProxy {
/// FavouritesState(...).copyWith(id: 12, name: "My name") /// FavouritesState(...).copyWith(id: 12, name: "My name")
/// ```` /// ````
FavouritesState call({ FavouritesState call({
List<String>? likedIds, List<String>? favouritesIds,
bool? hasFavouritesLoaded, bool? hasFavouritesLoaded,
}); });
} }
@ -30,7 +30,8 @@ class _$FavouritesStateCWProxyImpl implements _$FavouritesStateCWProxy {
final FavouritesState _value; final FavouritesState _value;
@override @override
FavouritesState likedIds(List<String>? likedIds) => this(likedIds: likedIds); FavouritesState favouritesIds(List<String>? favouritesIds) =>
this(favouritesIds: favouritesIds);
@override @override
FavouritesState hasFavouritesLoaded(bool hasFavouritesLoaded) => FavouritesState hasFavouritesLoaded(bool hasFavouritesLoaded) =>
@ -45,14 +46,14 @@ class _$FavouritesStateCWProxyImpl implements _$FavouritesStateCWProxy {
/// FavouritesState(...).copyWith(id: 12, name: "My name") /// FavouritesState(...).copyWith(id: 12, name: "My name")
/// ```` /// ````
FavouritesState call({ FavouritesState call({
Object? likedIds = const $CopyWithPlaceholder(), Object? favouritesIds = const $CopyWithPlaceholder(),
Object? hasFavouritesLoaded = const $CopyWithPlaceholder(), Object? hasFavouritesLoaded = const $CopyWithPlaceholder(),
}) { }) {
return FavouritesState( return FavouritesState(
favouritesIds: likedIds == const $CopyWithPlaceholder() favouritesIds: favouritesIds == const $CopyWithPlaceholder()
? _value.favouritesIds ? _value.favouritesIds
// ignore: cast_nullable_to_non_nullable // ignore: cast_nullable_to_non_nullable
: likedIds as List<String>?, : favouritesIds as List<String>?,
hasFavouritesLoaded: hasFavouritesLoaded:
hasFavouritesLoaded == const $CopyWithPlaceholder() || hasFavouritesLoaded == const $CopyWithPlaceholder() ||
hasFavouritesLoaded == null hasFavouritesLoaded == null

View File

@ -1,5 +1,8 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_android_app/components/extensions/context_x.dart'; import 'package:flutter_android_app/components/extensions/context_x.dart';
import 'package:flutter_android_app/presentation/currency_bloc/currency_bloc.dart';
import 'package:flutter_android_app/presentation/currency_bloc/currency_events.dart';
import 'package:flutter_android_app/presentation/currency_bloc/currency_state.dart';
import 'package:flutter_android_app/presentation/favourites_bloc/favourites_bloc.dart'; import 'package:flutter_android_app/presentation/favourites_bloc/favourites_bloc.dart';
import 'package:flutter_android_app/presentation/favourites_bloc/favourites_events.dart'; import 'package:flutter_android_app/presentation/favourites_bloc/favourites_events.dart';
import 'package:flutter_android_app/presentation/favourites_bloc/favourites_state.dart'; import 'package:flutter_android_app/presentation/favourites_bloc/favourites_state.dart';
@ -21,9 +24,12 @@ class FavouritesPage extends StatefulWidget {
class _FavouritesPageState extends State<FavouritesPage> { class _FavouritesPageState extends State<FavouritesPage> {
final ScrollController scrollController = ScrollController(); final ScrollController scrollController = ScrollController();
List<String>? favouritesIds; List<String>? favouritesIds;
bool wereIdsPreviouslyLoaded = false; bool wereIdsPreviouslyLoaded = false;
String? currencyId;
@override @override
void initState() { void initState() {
WidgetsBinding.instance.addPostFrameCallback((_) { WidgetsBinding.instance.addPostFrameCallback((_) {
@ -47,41 +53,46 @@ class _FavouritesPageState extends State<FavouritesPage> {
if (state.hasFavouritesLoaded && !wereIdsPreviouslyLoaded) { if (state.hasFavouritesLoaded && !wereIdsPreviouslyLoaded) {
wereIdsPreviouslyLoaded = true; wereIdsPreviouslyLoaded = true;
favouritesIds = state.favouritesIds; favouritesIds = state.favouritesIds;
context.read<HomeBloc>().add(HomeLoadFavouritesDataEvent( context.read<CurrencyBloc>().add(const LoadLocalCurrencyEvent());
ids: favouritesIds,
locale: context.locale,
));
} }
}, },
child: Padding( child: BlocListener<CurrencyBloc, CurrencyState>(
padding: EdgeInsets.only(top: 12), listener: (context, state) {
child: Column( if (state.hasCurrencyLoaded && state.currencyId != null) {
children: [ currencyId = state.currencyId;
CardsList( context.read<HomeBloc>().add(HomeLoadFavouritesDataEvent(
onListRefresh: _onRefresh, ids: favouritesIds,
onCardLiked: _onLike, locale: context.locale,
onCardTapped: _navToDetails, currencyId: currencyId!,
onNextPage: _onNextPage, ));
), }
BlocBuilder<HomeBloc, HomeState>( },
builder: (context, state) => state.isPaginationLoading child: Padding(
? const Padding( padding: const EdgeInsets.only(top: 12),
padding: EdgeInsets.all(12), child: Column(
child: CircularProgressIndicator(), children: [
) CardsList(
: const SizedBox.shrink(), onListRefresh: _onRefresh,
), onCardLiked: _onLike,
], onCardTapped: _navToDetails,
onNextPage: _onNextPage,
),
BlocBuilder<HomeBloc, HomeState>(
builder: (context, state) => state.isPaginationLoading
? const Padding(
padding: EdgeInsets.all(12),
child: CircularProgressIndicator(),
)
: const SizedBox.shrink(),
),
],
),
), ),
), ),
); );
} }
Future<void> _onRefresh() { Future<void> _onRefresh() {
//context.read<HomeBloc>().add(HomeLoadFavouritesDataEvent(
// ids: favouritesIds,
// locale: context.locale,
//));
wereIdsPreviouslyLoaded = false; wereIdsPreviouslyLoaded = false;
context.read<FavouritesBloc>().add(const LoadFavouritesEvent()); context.read<FavouritesBloc>().add(const LoadFavouritesEvent());
return Future.value(null); return Future.value(null);
@ -120,6 +131,7 @@ class _FavouritesPageState extends State<FavouritesPage> {
ids: favouritesIds, ids: favouritesIds,
nextPage: bloc.state.data?.nextPage, nextPage: bloc.state.data?.nextPage,
locale: context.locale, locale: context.locale,
currencyId: currencyId!,
)); ));
} }
} }

View File

@ -15,7 +15,10 @@ class HomeBloc extends Bloc<HomeEvent, HomeState> {
const int pageSize = 20; const int pageSize = 20;
if (event.nextPage == null) { if (event.nextPage == null) {
emit(state.copyWith(isLoading: true)); emit(state.copyWith(
isLoading: true,
error: 'NO_ERROR',
));
} else { } else {
emit(state.copyWith(isPaginationLoading: true)); emit(state.copyWith(isPaginationLoading: true));
} }
@ -23,11 +26,12 @@ class HomeBloc extends Bloc<HomeEvent, HomeState> {
String? error; String? error;
final data = await repo.loadData( final data = await repo.loadData(
search: event.search, search: event.search,
page: event.nextPage ?? 1, page: event.nextPage ?? 1,
pageSize: pageSize, pageSize: pageSize,
onError: (e) => error = e, onError: (e) => error = e,
locale: event.locale, locale: event.locale,
currencyId: event.currencyId,
); );
bool isLastPage = false; bool isLastPage = false;
@ -52,7 +56,10 @@ class HomeBloc extends Bloc<HomeEvent, HomeState> {
const int pageSize = 10; const int pageSize = 10;
if (event.nextPage == null) { if (event.nextPage == null) {
emit(state.copyWith(isLoading: true)); emit(state.copyWith(
isLoading: true,
error: 'NO_ERROR',
));
} else { } else {
emit(state.copyWith(isPaginationLoading: true)); emit(state.copyWith(isPaginationLoading: true));
} }
@ -65,6 +72,7 @@ class HomeBloc extends Bloc<HomeEvent, HomeState> {
pageSize: pageSize, pageSize: pageSize,
onError: (e) => error = e, onError: (e) => error = e,
locale: event.locale, locale: event.locale,
currencyId: event.currencyId,
); );
bool isLastPage = false; bool isLastPage = false;

View File

@ -8,14 +8,26 @@ class HomeLoadDataEvent extends HomeEvent {
final String? search; final String? search;
final int? nextPage; final int? nextPage;
final AppLocale? locale; final AppLocale? locale;
final String currencyId;
const HomeLoadDataEvent({this.search, this.nextPage, this.locale}); const HomeLoadDataEvent({
this.search,
this.nextPage,
this.locale,
required this.currencyId,
});
} }
class HomeLoadFavouritesDataEvent extends HomeEvent { class HomeLoadFavouritesDataEvent extends HomeEvent {
final List<String>? ids; final List<String>? ids;
final int? nextPage; final int? nextPage;
final AppLocale? locale; final AppLocale? locale;
final String currencyId;
const HomeLoadFavouritesDataEvent({this.ids, this.nextPage, this.locale}); const HomeLoadFavouritesDataEvent({
this.ids,
this.nextPage,
this.locale,
required this.currencyId,
});
} }

View File

@ -1,4 +1,5 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_android_app/components/extensions/context_x.dart';
import 'package:flutter_android_app/domain/models/card.dart'; import 'package:flutter_android_app/domain/models/card.dart';
import 'package:flutter_android_app/presentation/home_page/card_crypto.dart'; import 'package:flutter_android_app/presentation/home_page/card_crypto.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
@ -45,40 +46,63 @@ class _CardsListState extends State<CardsList> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return BlocBuilder<HomeBloc, HomeState>( return BlocListener<HomeBloc, HomeState>(
builder: (context, state) => state.error != null listener: (context, state) {
? Text( if (state.error != null && state.error != 'NO_ERROR') {
state.error ?? '', WidgetsBinding.instance.addPostFrameCallback((_) {
style: Theme.of(context).textTheme.headlineSmall?.copyWith(color: Colors.red), ScaffoldMessenger.of(context).showSnackBar(SnackBar(
) content: Text(
: state.isLoading context.locale.cardsLoadingFailedSnackBar,
? const Center(child: Padding( style: Theme.of(context).textTheme.bodyLarge,
padding: EdgeInsets.only(top: 20), ),
child: CircularProgressIndicator(), backgroundColor: Theme.of(context).colorScheme.errorContainer,
)) duration: const Duration(seconds: 2),
: BlocBuilder<FavouritesBloc, FavouritesState>( ));
builder: (context, likeState) => Expanded( });
child: RefreshIndicator( }
onRefresh: widget.onListRefresh, },
child: ListView.builder( child: BlocBuilder<HomeBloc, HomeState>(
controller: scrollController, builder: (context, state) => state.error != null && state.error != 'NO_ERROR'
padding: EdgeInsets.zero, ? Center(
itemCount: state.data?.data?.length ?? 0, child: Padding(
itemBuilder: (context, index) { padding: const EdgeInsets.all(20),
final data = state.data?.data?[index]; child: Text(
return data != null context.locale.cardsLoadingFailed,
? CardCrypto.fromData( style: Theme.of(context).textTheme.headlineSmall?.copyWith(color: Theme.of(context).disabledColor),
data, ),
isLiked: likeState.favouritesIds?.contains(data.id) == true, ),
onLike: widget.onCardLiked, )
onTap: () => widget.onCardTapped?.call(data), : state.isLoading
) ? const Center(
: const SizedBox.shrink(); child: Padding(
}, padding: EdgeInsets.only(top: 20),
child: CircularProgressIndicator(),
)
)
: BlocBuilder<FavouritesBloc, FavouritesState>(
builder: (context, likeState) => Expanded(
child: RefreshIndicator(
onRefresh: widget.onListRefresh,
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
? CardCrypto.fromData(
data,
isLiked: likeState.favouritesIds?.contains(data.id) == true,
onLike: widget.onCardLiked,
onTap: () => widget.onCardTapped?.call(data),
)
: const SizedBox.shrink();
},
),
), ),
), ),
), )
) ),
); );
} }

View File

@ -11,6 +11,9 @@ import 'package:flutter_android_app/presentation/home_page/cards_list.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import '../common/svg_objects.dart'; import '../common/svg_objects.dart';
import '../currency_bloc/currency_bloc.dart';
import '../currency_bloc/currency_events.dart';
import '../currency_bloc/currency_state.dart';
import '../favourites_bloc/favourites_bloc.dart'; import '../favourites_bloc/favourites_bloc.dart';
import '../favourites_bloc/favourites_events.dart'; import '../favourites_bloc/favourites_events.dart';
import '../settings_page/settings_page.dart'; import '../settings_page/settings_page.dart';
@ -83,13 +86,14 @@ class HomePage extends StatefulWidget {
class _HomePageState extends State<HomePage> { class _HomePageState extends State<HomePage> {
final TextEditingController searchController = TextEditingController(); final TextEditingController searchController = TextEditingController();
late String? currencyId;
@override @override
void initState() { void initState() {
SvgObjects.init(); SvgObjects.init();
WidgetsBinding.instance.addPostFrameCallback((_) { WidgetsBinding.instance.addPostFrameCallback((_) {
context.read<HomeBloc>().add(HomeLoadDataEvent(locale: context.locale)); context.read<CurrencyBloc>().add(const LoadLocalCurrencyEvent());
context.read<FavouritesBloc>().add(const LoadFavouritesEvent()); context.read<FavouritesBloc>().add(const LoadFavouritesEvent());
}); });
@ -105,51 +109,70 @@ class _HomePageState extends State<HomePage> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Padding( return BlocListener<CurrencyBloc, CurrencyState>(
padding: EdgeInsets.only(top: MediaQuery.of(context).padding.top), listener: (context, state) {
child: Column( if (state.hasCurrencyLoaded && state.currencyId != null) {
children: [ currencyId = state.currencyId;
Padding( context.read<HomeBloc>().add(HomeLoadDataEvent(
padding: const EdgeInsets.all(12), locale: context.locale,
child: SearchBar( currencyId: currencyId!,
controller: searchController, ));
onChanged: (search) { }
Debounce.run(() => context.read<HomeBloc>().add(HomeLoadDataEvent(search: search, locale: context.locale))); },
}, child: Padding(
leading: const Icon(Icons.search), padding: EdgeInsets.only(top: MediaQuery.of(context).padding.top),
trailing: [ child: Column(
IconButton( children: [
icon: const Icon(Icons.close), Padding(
onPressed: () { padding: const EdgeInsets.all(12),
if (searchController.text.isNotEmpty) { child: SearchBar(
searchController.clear(); controller: searchController,
context.read<HomeBloc>().add(HomeLoadDataEvent(locale: context.locale)); onChanged: (search) {
} Debounce.run(() => context.read<HomeBloc>().add(HomeLoadDataEvent(
}, search: search,
), locale: context.locale,
], currencyId: currencyId!,
hintText: context.locale.searchHint, )));
elevation: const WidgetStatePropertyAll(0.0), },
padding: const WidgetStatePropertyAll(EdgeInsets.only(left: 18, right: 10)), leading: const Icon(Icons.search),
backgroundColor: WidgetStatePropertyAll(Theme.of(context).colorScheme.secondaryContainer), trailing: [
IconButton(
icon: const Icon(Icons.close),
onPressed: () {
if (searchController.text.isNotEmpty) {
FocusManager.instance.primaryFocus?.unfocus();
searchController.clear();
context.read<HomeBloc>().add(HomeLoadDataEvent(
locale: context.locale,
currencyId: currencyId!,
));
}
},
),
],
hintText: context.locale.searchHint,
elevation: const WidgetStatePropertyAll(0.0),
padding: const WidgetStatePropertyAll(EdgeInsets.only(left: 18, right: 10)),
backgroundColor: WidgetStatePropertyAll(Theme.of(context).colorScheme.secondaryContainer),
),
), ),
), CardsList(
CardsList( onListRefresh: _onRefresh,
onListRefresh: _onRefresh, onCardLiked: _onLike,
onCardLiked: _onLike, onCardTapped: _navToDetails,
onCardTapped: _navToDetails, onNextPage: _onNextPage,
onNextPage: _onNextPage, ),
), BlocBuilder<HomeBloc, HomeState>(
BlocBuilder<HomeBloc, HomeState>( builder: (context, state) => state.isPaginationLoading
builder: (context, state) => state.isPaginationLoading ? const Padding(
? const Padding( padding: EdgeInsets.all(12),
padding: EdgeInsets.all(12), child: CircularProgressIndicator(),
child: CircularProgressIndicator(), )
) : const SizedBox.shrink(),
: const SizedBox.shrink(), ),
), ],
], )
) ),
); );
} }
@ -167,7 +190,11 @@ class _HomePageState extends State<HomePage> {
} }
Future<void> _onRefresh() { Future<void> _onRefresh() {
context.read<HomeBloc>().add(HomeLoadDataEvent(search: searchController.text, locale: context.locale)); context.read<HomeBloc>().add(HomeLoadDataEvent(
search: searchController.text,
locale: context.locale,
currencyId: currencyId!,
));
return Future.value(null); return Future.value(null);
} }
@ -179,6 +206,7 @@ class _HomePageState extends State<HomePage> {
} }
void _navToDetails(CardData data) { void _navToDetails(CardData data) {
FocusManager.instance.primaryFocus?.unfocus();
Navigator.push(context, MaterialPageRoute<void>( Navigator.push(context, MaterialPageRoute<void>(
builder: (context) => DetailsPage(data)), builder: (context) => DetailsPage(data)),
); );
@ -191,6 +219,7 @@ class _HomePageState extends State<HomePage> {
search: searchController.text, search: searchController.text,
nextPage: bloc.state.data?.nextPage, nextPage: bloc.state.data?.nextPage,
locale: context.locale, locale: context.locale,
currencyId: currencyId!,
)); ));
} }
} }

View File

@ -1,5 +1,8 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_android_app/components/extensions/context_x.dart'; import 'package:flutter_android_app/components/extensions/context_x.dart';
import 'package:flutter_android_app/presentation/currency_bloc/currency_bloc.dart';
import 'package:flutter_android_app/presentation/currency_bloc/currency_events.dart';
import 'package:flutter_android_app/presentation/currency_bloc/currency_state.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import '../common/svg_objects.dart'; import '../common/svg_objects.dart';
@ -40,6 +43,34 @@ class SettingsPage extends StatelessWidget {
), ),
], ],
), ),
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'${context.locale.settingsCurrency}:',
style: Theme.of(context).textTheme.titleMedium,
),
BlocBuilder<CurrencyBloc, CurrencyState>(
builder: (context, currencyState) => GestureDetector(
onTap: () => context.read<CurrencyBloc>().add(const ToggleLocalCurrencyEvent()),
child: SizedBox.square(
dimension: 50,
child: Padding(
padding: const EdgeInsets.only(right: 0),
child: BlocBuilder<LocaleBloc, LocaleState>(
builder: (context, state) {
return switch (currencyState.currencyId) {
'rub' => const SvgRu(),
'usd' => const SvgUs(),
_ => const SvgUs(),
};
}),
),
),
),
),
],
),
], ],
), ),
); );

View File

@ -10,6 +10,7 @@ abstract class ApiInterface {
int page = 1, int page = 1,
int pageSize = 20, int pageSize = 20,
AppLocale? locale, AppLocale? locale,
String currencyId,
}); });
Future<HomeData?> loadDataWithIds({ Future<HomeData?> loadDataWithIds({
@ -18,5 +19,6 @@ abstract class ApiInterface {
int page = 1, int page = 1,
int pageSize = 8, int pageSize = 8,
AppLocale? locale, AppLocale? locale,
String currencyId,
}); });
} }

View File

@ -29,11 +29,12 @@ class CryptoRepository extends ApiInterface {
int page = 1, int page = 1,
int pageSize = 20, int pageSize = 20,
AppLocale? locale, AppLocale? locale,
String currencyId = '',
}) async { }) async {
try { try {
Map<String, dynamic> queryParams = { Map<String, dynamic> queryParams = {
'x_cg_demo_api_key': _apiKey, 'x_cg_demo_api_key': _apiKey,
'vs_currency': _getCurrencyName(locale?.localeName), 'vs_currency': currencyId,
'per_page': pageSize, 'per_page': pageSize,
'page': page, 'page': page,
}; };
@ -68,7 +69,7 @@ class CryptoRepository extends ApiInterface {
); );
final CoinsDto dto = CoinsDto.fromJson(response.data as List<dynamic>); final CoinsDto dto = CoinsDto.fromJson(response.data as List<dynamic>);
final HomeData data = dto.toDomain(locale, page); final HomeData data = dto.toDomain(locale, currencyId, page);
return data; return data;
} on DioException catch (e) { } on DioException catch (e) {
@ -84,11 +85,12 @@ class CryptoRepository extends ApiInterface {
int page = 1, int page = 1,
int pageSize = 20, int pageSize = 20,
AppLocale? locale, AppLocale? locale,
String currencyId = '',
}) async { }) async {
try { try {
Map<String, dynamic> queryParams = { Map<String, dynamic> queryParams = {
'x_cg_demo_api_key': _apiKey, 'x_cg_demo_api_key': _apiKey,
'vs_currency': _getCurrencyName(locale?.localeName), 'vs_currency': currencyId,
'per_page': pageSize, 'per_page': pageSize,
'page': page, 'page': page,
}; };
@ -108,7 +110,7 @@ class CryptoRepository extends ApiInterface {
); );
final CoinsDto dto = CoinsDto.fromJson(response.data as List<dynamic>); final CoinsDto dto = CoinsDto.fromJson(response.data as List<dynamic>);
final HomeData data = dto.toDomain(locale, page); final HomeData data = dto.toDomain(locale, currencyId, page);
return data; return data;
} on DioException catch (e) { } on DioException catch (e) {
@ -116,15 +118,4 @@ class CryptoRepository extends ApiInterface {
return null; return null;
} }
} }
String _getCurrencyName(String? localeName) {
if (localeName == null) {
return 'usd';
}
return switch (localeName) {
'ru' => 'rub',
_ => 'usd',
};
}
} }

View File

@ -13,6 +13,7 @@ class MockRepository extends ApiInterface {
int page = 1, int page = 1,
int pageSize = 20, int pageSize = 20,
AppLocale? locale, AppLocale? locale,
String currencyId = '',
}) async { }) async {
return HomeData(data: [ return HomeData(data: [
CardData( CardData(
@ -46,6 +47,7 @@ class MockRepository extends ApiInterface {
int page = 1, int page = 1,
int pageSize = 20, int pageSize = 20,
AppLocale? locale, AppLocale? locale,
String currencyId = '',
}) async { }) async {
return HomeData(data: [ return HomeData(data: [
CardData( CardData(