This commit is contained in:
Phuoc Nguyen
2025-10-03 17:54:39 +07:00
parent 762395ce50
commit 38a33743e6
16 changed files with 1096 additions and 228 deletions

View File

@@ -0,0 +1,28 @@
import 'package:base_flutter/core/utils/utils.dart';
import '../../../../core/network/dio_client.dart';
import '../../../../core/constants/environment_config.dart';
import '../../../../core/services/api_service.dart';
import '../models/todo_model.dart';
abstract class TodoRemoteDataSource {
Future<List<TodoModel>> getTodos();
}
class TodoRemoteDataSourceImpl extends BaseApiService
implements TodoRemoteDataSource {
TodoRemoteDataSourceImpl({required DioClient dioClient}) : super(dioClient);
@override
Future<List<TodoModel>> getTodos() async {
final response = await dioClient.get(EnvironmentConfig.todosEndpoint);
if (response.data is List) {
final List<dynamic> todosJson = response.data as List<dynamic>;
return todosJson
.map((json) => TodoModel.fromJson(json as DataMap))
.toList();
} else {
throw Exception('Expected List but got ${response.data.runtimeType}');
}
}
}

View File

@@ -0,0 +1,46 @@
import 'package:freezed_annotation/freezed_annotation.dart';
import '../../domain/entities/todo.dart';
part 'todo_model.freezed.dart';
part 'todo_model.g.dart';
@freezed
class TodoModel with _$TodoModel {
const factory TodoModel({
required int id,
required String title,
String? description,
required bool completed,
required String userId,
@JsonKey(includeFromJson: false, includeToJson: false) Map<String, dynamic>? user,
DateTime? createdAt,
DateTime? updatedAt,
}) = _TodoModel;
const TodoModel._();
factory TodoModel.fromJson(Map<String, dynamic> json) =>
_$TodoModelFromJson(json);
/// Convert to domain entity
Todo toEntity() => Todo(
id: id,
title: title,
description: description,
completed: completed,
userId: userId,
createdAt: createdAt,
updatedAt: updatedAt,
);
/// Create from domain entity
factory TodoModel.fromEntity(Todo todo) => TodoModel(
id: todo.id,
title: todo.title,
description: todo.description,
completed: todo.completed,
userId: todo.userId,
createdAt: todo.createdAt,
updatedAt: todo.updatedAt,
);
}

View File

@@ -0,0 +1,325 @@
// coverage:ignore-file
// GENERATED CODE - DO NOT MODIFY BY HAND
// ignore_for_file: type=lint
// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark
part of 'todo_model.dart';
// **************************************************************************
// FreezedGenerator
// **************************************************************************
T _$identity<T>(T value) => value;
final _privateConstructorUsedError = UnsupportedError(
'It seems like you constructed your class using `MyClass._()`. This constructor is only meant to be used by freezed and you are not supposed to need it nor use it.\nPlease check the documentation here for more information: https://github.com/rrousselGit/freezed#adding-getters-and-methods-to-our-models');
TodoModel _$TodoModelFromJson(Map<String, dynamic> json) {
return _TodoModel.fromJson(json);
}
/// @nodoc
mixin _$TodoModel {
int get id => throw _privateConstructorUsedError;
String get title => throw _privateConstructorUsedError;
String? get description => throw _privateConstructorUsedError;
bool get completed => throw _privateConstructorUsedError;
String get userId => throw _privateConstructorUsedError;
@JsonKey(includeFromJson: false, includeToJson: false)
Map<String, dynamic>? get user => throw _privateConstructorUsedError;
DateTime? get createdAt => throw _privateConstructorUsedError;
DateTime? get updatedAt => throw _privateConstructorUsedError;
Map<String, dynamic> toJson() => throw _privateConstructorUsedError;
@JsonKey(ignore: true)
$TodoModelCopyWith<TodoModel> get copyWith =>
throw _privateConstructorUsedError;
}
/// @nodoc
abstract class $TodoModelCopyWith<$Res> {
factory $TodoModelCopyWith(TodoModel value, $Res Function(TodoModel) then) =
_$TodoModelCopyWithImpl<$Res, TodoModel>;
@useResult
$Res call(
{int id,
String title,
String? description,
bool completed,
String userId,
@JsonKey(includeFromJson: false, includeToJson: false)
Map<String, dynamic>? user,
DateTime? createdAt,
DateTime? updatedAt});
}
/// @nodoc
class _$TodoModelCopyWithImpl<$Res, $Val extends TodoModel>
implements $TodoModelCopyWith<$Res> {
_$TodoModelCopyWithImpl(this._value, this._then);
// ignore: unused_field
final $Val _value;
// ignore: unused_field
final $Res Function($Val) _then;
@pragma('vm:prefer-inline')
@override
$Res call({
Object? id = null,
Object? title = null,
Object? description = freezed,
Object? completed = null,
Object? userId = null,
Object? user = freezed,
Object? createdAt = freezed,
Object? updatedAt = freezed,
}) {
return _then(_value.copyWith(
id: null == id
? _value.id
: id // ignore: cast_nullable_to_non_nullable
as int,
title: null == title
? _value.title
: title // ignore: cast_nullable_to_non_nullable
as String,
description: freezed == description
? _value.description
: description // ignore: cast_nullable_to_non_nullable
as String?,
completed: null == completed
? _value.completed
: completed // ignore: cast_nullable_to_non_nullable
as bool,
userId: null == userId
? _value.userId
: userId // ignore: cast_nullable_to_non_nullable
as String,
user: freezed == user
? _value.user
: user // ignore: cast_nullable_to_non_nullable
as Map<String, dynamic>?,
createdAt: freezed == createdAt
? _value.createdAt
: createdAt // ignore: cast_nullable_to_non_nullable
as DateTime?,
updatedAt: freezed == updatedAt
? _value.updatedAt
: updatedAt // ignore: cast_nullable_to_non_nullable
as DateTime?,
) as $Val);
}
}
/// @nodoc
abstract class _$$TodoModelImplCopyWith<$Res>
implements $TodoModelCopyWith<$Res> {
factory _$$TodoModelImplCopyWith(
_$TodoModelImpl value, $Res Function(_$TodoModelImpl) then) =
__$$TodoModelImplCopyWithImpl<$Res>;
@override
@useResult
$Res call(
{int id,
String title,
String? description,
bool completed,
String userId,
@JsonKey(includeFromJson: false, includeToJson: false)
Map<String, dynamic>? user,
DateTime? createdAt,
DateTime? updatedAt});
}
/// @nodoc
class __$$TodoModelImplCopyWithImpl<$Res>
extends _$TodoModelCopyWithImpl<$Res, _$TodoModelImpl>
implements _$$TodoModelImplCopyWith<$Res> {
__$$TodoModelImplCopyWithImpl(
_$TodoModelImpl _value, $Res Function(_$TodoModelImpl) _then)
: super(_value, _then);
@pragma('vm:prefer-inline')
@override
$Res call({
Object? id = null,
Object? title = null,
Object? description = freezed,
Object? completed = null,
Object? userId = null,
Object? user = freezed,
Object? createdAt = freezed,
Object? updatedAt = freezed,
}) {
return _then(_$TodoModelImpl(
id: null == id
? _value.id
: id // ignore: cast_nullable_to_non_nullable
as int,
title: null == title
? _value.title
: title // ignore: cast_nullable_to_non_nullable
as String,
description: freezed == description
? _value.description
: description // ignore: cast_nullable_to_non_nullable
as String?,
completed: null == completed
? _value.completed
: completed // ignore: cast_nullable_to_non_nullable
as bool,
userId: null == userId
? _value.userId
: userId // ignore: cast_nullable_to_non_nullable
as String,
user: freezed == user
? _value._user
: user // ignore: cast_nullable_to_non_nullable
as Map<String, dynamic>?,
createdAt: freezed == createdAt
? _value.createdAt
: createdAt // ignore: cast_nullable_to_non_nullable
as DateTime?,
updatedAt: freezed == updatedAt
? _value.updatedAt
: updatedAt // ignore: cast_nullable_to_non_nullable
as DateTime?,
));
}
}
/// @nodoc
@JsonSerializable()
class _$TodoModelImpl extends _TodoModel {
const _$TodoModelImpl(
{required this.id,
required this.title,
this.description,
required this.completed,
required this.userId,
@JsonKey(includeFromJson: false, includeToJson: false)
final Map<String, dynamic>? user,
this.createdAt,
this.updatedAt})
: _user = user,
super._();
factory _$TodoModelImpl.fromJson(Map<String, dynamic> json) =>
_$$TodoModelImplFromJson(json);
@override
final int id;
@override
final String title;
@override
final String? description;
@override
final bool completed;
@override
final String userId;
final Map<String, dynamic>? _user;
@override
@JsonKey(includeFromJson: false, includeToJson: false)
Map<String, dynamic>? get user {
final value = _user;
if (value == null) return null;
if (_user is EqualUnmodifiableMapView) return _user;
// ignore: implicit_dynamic_type
return EqualUnmodifiableMapView(value);
}
@override
final DateTime? createdAt;
@override
final DateTime? updatedAt;
@override
String toString() {
return 'TodoModel(id: $id, title: $title, description: $description, completed: $completed, userId: $userId, user: $user, createdAt: $createdAt, updatedAt: $updatedAt)';
}
@override
bool operator ==(Object other) {
return identical(this, other) ||
(other.runtimeType == runtimeType &&
other is _$TodoModelImpl &&
(identical(other.id, id) || other.id == id) &&
(identical(other.title, title) || other.title == title) &&
(identical(other.description, description) ||
other.description == description) &&
(identical(other.completed, completed) ||
other.completed == completed) &&
(identical(other.userId, userId) || other.userId == userId) &&
const DeepCollectionEquality().equals(other._user, _user) &&
(identical(other.createdAt, createdAt) ||
other.createdAt == createdAt) &&
(identical(other.updatedAt, updatedAt) ||
other.updatedAt == updatedAt));
}
@JsonKey(ignore: true)
@override
int get hashCode => Object.hash(
runtimeType,
id,
title,
description,
completed,
userId,
const DeepCollectionEquality().hash(_user),
createdAt,
updatedAt);
@JsonKey(ignore: true)
@override
@pragma('vm:prefer-inline')
_$$TodoModelImplCopyWith<_$TodoModelImpl> get copyWith =>
__$$TodoModelImplCopyWithImpl<_$TodoModelImpl>(this, _$identity);
@override
Map<String, dynamic> toJson() {
return _$$TodoModelImplToJson(
this,
);
}
}
abstract class _TodoModel extends TodoModel {
const factory _TodoModel(
{required final int id,
required final String title,
final String? description,
required final bool completed,
required final String userId,
@JsonKey(includeFromJson: false, includeToJson: false)
final Map<String, dynamic>? user,
final DateTime? createdAt,
final DateTime? updatedAt}) = _$TodoModelImpl;
const _TodoModel._() : super._();
factory _TodoModel.fromJson(Map<String, dynamic> json) =
_$TodoModelImpl.fromJson;
@override
int get id;
@override
String get title;
@override
String? get description;
@override
bool get completed;
@override
String get userId;
@override
@JsonKey(includeFromJson: false, includeToJson: false)
Map<String, dynamic>? get user;
@override
DateTime? get createdAt;
@override
DateTime? get updatedAt;
@override
@JsonKey(ignore: true)
_$$TodoModelImplCopyWith<_$TodoModelImpl> get copyWith =>
throw _privateConstructorUsedError;
}

View File

@@ -0,0 +1,33 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'todo_model.dart';
// **************************************************************************
// JsonSerializableGenerator
// **************************************************************************
_$TodoModelImpl _$$TodoModelImplFromJson(Map<String, dynamic> json) =>
_$TodoModelImpl(
id: (json['id'] as num).toInt(),
title: json['title'] as String,
description: json['description'] as String?,
completed: json['completed'] as bool,
userId: json['userId'] as String,
createdAt: json['createdAt'] == null
? null
: DateTime.parse(json['createdAt'] as String),
updatedAt: json['updatedAt'] == null
? null
: DateTime.parse(json['updatedAt'] as String),
);
Map<String, dynamic> _$$TodoModelImplToJson(_$TodoModelImpl instance) =>
<String, dynamic>{
'id': instance.id,
'title': instance.title,
'description': instance.description,
'completed': instance.completed,
'userId': instance.userId,
'createdAt': instance.createdAt?.toIso8601String(),
'updatedAt': instance.updatedAt?.toIso8601String(),
};

View File

@@ -0,0 +1,49 @@
import 'package:fpdart/fpdart.dart';
import '../../../../core/errors/exceptions.dart';
import '../../../../core/errors/failures.dart';
import '../../../../core/network/network_info.dart';
import '../../domain/entities/todo.dart';
import '../../domain/repositories/todo_repository.dart';
import '../datasources/todo_remote_datasource.dart';
class TodoRepositoryImpl implements TodoRepository {
final TodoRemoteDataSource remoteDataSource;
final NetworkInfo networkInfo;
TodoRepositoryImpl({
required this.remoteDataSource,
required this.networkInfo,
});
@override
Future<Either<Failure, List<Todo>>> getTodos() async {
if (!await networkInfo.isConnected) {
return const Left(NetworkFailure('No internet connection'));
}
try {
final todos = await remoteDataSource.getTodos();
return Right(todos.map((model) => model.toEntity()).toList());
} on ServerException catch (e) {
return Left(ServerFailure(e.message));
} catch (e) {
return Left(ServerFailure(e.toString()));
}
}
@override
Future<Either<Failure, void>> refreshTodos() async {
if (!await networkInfo.isConnected) {
return const Left(NetworkFailure('No internet connection'));
}
try {
await remoteDataSource.getTodos();
return const Right(null);
} on ServerException catch (e) {
return Left(ServerFailure(e.message));
} catch (e) {
return Left(ServerFailure(e.toString()));
}
}
}

View File

@@ -0,0 +1,44 @@
import 'package:equatable/equatable.dart';
class Todo extends Equatable {
final int id;
final String title;
final String? description;
final bool completed;
final String userId;
final DateTime? createdAt;
final DateTime? updatedAt;
const Todo({
required this.id,
required this.title,
this.description,
required this.completed,
required this.userId,
this.createdAt,
this.updatedAt,
});
Todo copyWith({
int? id,
String? title,
String? description,
bool? completed,
String? userId,
DateTime? createdAt,
DateTime? updatedAt,
}) {
return Todo(
id: id ?? this.id,
title: title ?? this.title,
description: description ?? this.description,
completed: completed ?? this.completed,
userId: userId ?? this.userId,
createdAt: createdAt ?? this.createdAt,
updatedAt: updatedAt ?? this.updatedAt,
);
}
@override
List<Object?> get props => [id, title, description, completed, userId, createdAt, updatedAt];
}

View File

@@ -0,0 +1,8 @@
import 'package:fpdart/fpdart.dart';
import '../../../../core/errors/failures.dart';
import '../entities/todo.dart';
abstract class TodoRepository {
Future<Either<Failure, List<Todo>>> getTodos();
Future<Either<Failure, void>> refreshTodos();
}

View File

@@ -0,0 +1,112 @@
import 'package:riverpod_annotation/riverpod_annotation.dart';
import '../../../../core/providers/app_providers.dart';
import '../../../../core/providers/network_providers.dart';
import '../../data/datasources/todo_remote_datasource.dart';
import '../../data/repositories/todo_repository_impl.dart';
import '../../domain/entities/todo.dart';
import '../../domain/repositories/todo_repository.dart';
part 'todo_providers.g.dart';
/// Todo Remote DataSource Provider
@riverpod
TodoRemoteDataSource todoRemoteDataSource(TodoRemoteDataSourceRef ref) {
final dioClient = ref.watch(dioClientProvider);
return TodoRemoteDataSourceImpl(dioClient: dioClient);
}
/// Todo Repository Provider
@riverpod
TodoRepository todoRepository(TodoRepositoryRef ref) {
final remoteDataSource = ref.watch(todoRemoteDataSourceProvider);
final networkInfo = ref.watch(networkInfoProvider);
return TodoRepositoryImpl(
remoteDataSource: remoteDataSource,
networkInfo: networkInfo,
);
}
/// Todos State Provider - Fetches and manages todos list
@riverpod
class Todos extends _$Todos {
@override
Future<List<Todo>> build() async {
// Auto-fetch todos when provider is first accessed
return _fetchTodos();
}
Future<List<Todo>> _fetchTodos() async {
final repository = ref.read(todoRepositoryProvider);
final result = await repository.getTodos();
return result.fold(
(failure) => throw Exception(failure.message),
(todos) => todos,
);
}
/// Refresh todos from API
Future<void> refresh() async {
state = const AsyncValue.loading();
state = await AsyncValue.guard(() => _fetchTodos());
}
/// Toggle todo completion status (local only for now)
void toggleTodo(int id) {
state.whenData((todos) {
final updatedTodos = todos.map((todo) {
if (todo.id == id) {
return todo.copyWith(completed: !todo.completed);
}
return todo;
}).toList();
state = AsyncValue.data(updatedTodos);
});
}
}
/// Filtered Todos Provider - Filter todos by search query
@riverpod
List<Todo> filteredTodos(FilteredTodosRef ref, String searchQuery) {
final todosAsync = ref.watch(todosProvider);
return todosAsync.when(
data: (todos) {
if (searchQuery.isEmpty) {
return todos;
}
return todos.where((todo) {
return todo.title.toLowerCase().contains(searchQuery.toLowerCase()) ||
(todo.description?.toLowerCase().contains(searchQuery.toLowerCase()) ?? false);
}).toList();
},
loading: () => [],
error: (_, __) => [],
);
}
/// Completed Todos Count Provider
@riverpod
int completedTodosCount(CompletedTodosCountRef ref) {
final todosAsync = ref.watch(todosProvider);
return todosAsync.when(
data: (todos) => todos.where((todo) => todo.completed).length,
loading: () => 0,
error: (_, __) => 0,
);
}
/// Pending Todos Count Provider
@riverpod
int pendingTodosCount(PendingTodosCountRef ref) {
final todosAsync = ref.watch(todosProvider);
return todosAsync.when(
data: (todos) => todos.where((todo) => !todo.completed).length,
loading: () => 0,
error: (_, __) => 0,
);
}

View File

@@ -0,0 +1,259 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'todo_providers.dart';
// **************************************************************************
// RiverpodGenerator
// **************************************************************************
String _$todoRemoteDataSourceHash() =>
r'10f103aa6cd7de9c9829c3554f317065c7115575';
/// Todo Remote DataSource Provider
///
/// Copied from [todoRemoteDataSource].
@ProviderFor(todoRemoteDataSource)
final todoRemoteDataSourceProvider =
AutoDisposeProvider<TodoRemoteDataSource>.internal(
todoRemoteDataSource,
name: r'todoRemoteDataSourceProvider',
debugGetCreateSourceHash: const bool.fromEnvironment('dart.vm.product')
? null
: _$todoRemoteDataSourceHash,
dependencies: null,
allTransitiveDependencies: null,
);
typedef TodoRemoteDataSourceRef = AutoDisposeProviderRef<TodoRemoteDataSource>;
String _$todoRepositoryHash() => r'6830b5ede91b11ac04d0a9430cb84a0f2a8d0905';
/// Todo Repository Provider
///
/// Copied from [todoRepository].
@ProviderFor(todoRepository)
final todoRepositoryProvider = AutoDisposeProvider<TodoRepository>.internal(
todoRepository,
name: r'todoRepositoryProvider',
debugGetCreateSourceHash: const bool.fromEnvironment('dart.vm.product')
? null
: _$todoRepositoryHash,
dependencies: null,
allTransitiveDependencies: null,
);
typedef TodoRepositoryRef = AutoDisposeProviderRef<TodoRepository>;
String _$filteredTodosHash() => r'b814fe45ea117a5f71e9a223c39c2cfb5fcff61a';
/// Copied from Dart SDK
class _SystemHash {
_SystemHash._();
static int combine(int hash, int value) {
// ignore: parameter_assignments
hash = 0x1fffffff & (hash + value);
// ignore: parameter_assignments
hash = 0x1fffffff & (hash + ((0x0007ffff & hash) << 10));
return hash ^ (hash >> 6);
}
static int finish(int hash) {
// ignore: parameter_assignments
hash = 0x1fffffff & (hash + ((0x03ffffff & hash) << 3));
// ignore: parameter_assignments
hash = hash ^ (hash >> 11);
return 0x1fffffff & (hash + ((0x00003fff & hash) << 15));
}
}
/// Filtered Todos Provider - Filter todos by search query
///
/// Copied from [filteredTodos].
@ProviderFor(filteredTodos)
const filteredTodosProvider = FilteredTodosFamily();
/// Filtered Todos Provider - Filter todos by search query
///
/// Copied from [filteredTodos].
class FilteredTodosFamily extends Family<List<Todo>> {
/// Filtered Todos Provider - Filter todos by search query
///
/// Copied from [filteredTodos].
const FilteredTodosFamily();
/// Filtered Todos Provider - Filter todos by search query
///
/// Copied from [filteredTodos].
FilteredTodosProvider call(
String searchQuery,
) {
return FilteredTodosProvider(
searchQuery,
);
}
@override
FilteredTodosProvider getProviderOverride(
covariant FilteredTodosProvider provider,
) {
return call(
provider.searchQuery,
);
}
static const Iterable<ProviderOrFamily>? _dependencies = null;
@override
Iterable<ProviderOrFamily>? get dependencies => _dependencies;
static const Iterable<ProviderOrFamily>? _allTransitiveDependencies = null;
@override
Iterable<ProviderOrFamily>? get allTransitiveDependencies =>
_allTransitiveDependencies;
@override
String? get name => r'filteredTodosProvider';
}
/// Filtered Todos Provider - Filter todos by search query
///
/// Copied from [filteredTodos].
class FilteredTodosProvider extends AutoDisposeProvider<List<Todo>> {
/// Filtered Todos Provider - Filter todos by search query
///
/// Copied from [filteredTodos].
FilteredTodosProvider(
String searchQuery,
) : this._internal(
(ref) => filteredTodos(
ref as FilteredTodosRef,
searchQuery,
),
from: filteredTodosProvider,
name: r'filteredTodosProvider',
debugGetCreateSourceHash:
const bool.fromEnvironment('dart.vm.product')
? null
: _$filteredTodosHash,
dependencies: FilteredTodosFamily._dependencies,
allTransitiveDependencies:
FilteredTodosFamily._allTransitiveDependencies,
searchQuery: searchQuery,
);
FilteredTodosProvider._internal(
super._createNotifier, {
required super.name,
required super.dependencies,
required super.allTransitiveDependencies,
required super.debugGetCreateSourceHash,
required super.from,
required this.searchQuery,
}) : super.internal();
final String searchQuery;
@override
Override overrideWith(
List<Todo> Function(FilteredTodosRef provider) create,
) {
return ProviderOverride(
origin: this,
override: FilteredTodosProvider._internal(
(ref) => create(ref as FilteredTodosRef),
from: from,
name: null,
dependencies: null,
allTransitiveDependencies: null,
debugGetCreateSourceHash: null,
searchQuery: searchQuery,
),
);
}
@override
AutoDisposeProviderElement<List<Todo>> createElement() {
return _FilteredTodosProviderElement(this);
}
@override
bool operator ==(Object other) {
return other is FilteredTodosProvider && other.searchQuery == searchQuery;
}
@override
int get hashCode {
var hash = _SystemHash.combine(0, runtimeType.hashCode);
hash = _SystemHash.combine(hash, searchQuery.hashCode);
return _SystemHash.finish(hash);
}
}
mixin FilteredTodosRef on AutoDisposeProviderRef<List<Todo>> {
/// The parameter `searchQuery` of this provider.
String get searchQuery;
}
class _FilteredTodosProviderElement
extends AutoDisposeProviderElement<List<Todo>> with FilteredTodosRef {
_FilteredTodosProviderElement(super.provider);
@override
String get searchQuery => (origin as FilteredTodosProvider).searchQuery;
}
String _$completedTodosCountHash() =>
r'9905f3fbd8c17b4cd4edde44d34b36e7b4d1f582';
/// Completed Todos Count Provider
///
/// Copied from [completedTodosCount].
@ProviderFor(completedTodosCount)
final completedTodosCountProvider = AutoDisposeProvider<int>.internal(
completedTodosCount,
name: r'completedTodosCountProvider',
debugGetCreateSourceHash: const bool.fromEnvironment('dart.vm.product')
? null
: _$completedTodosCountHash,
dependencies: null,
allTransitiveDependencies: null,
);
typedef CompletedTodosCountRef = AutoDisposeProviderRef<int>;
String _$pendingTodosCountHash() => r'f302d2335102b191a27f5ad628d01f9d1cffea05';
/// Pending Todos Count Provider
///
/// Copied from [pendingTodosCount].
@ProviderFor(pendingTodosCount)
final pendingTodosCountProvider = AutoDisposeProvider<int>.internal(
pendingTodosCount,
name: r'pendingTodosCountProvider',
debugGetCreateSourceHash: const bool.fromEnvironment('dart.vm.product')
? null
: _$pendingTodosCountHash,
dependencies: null,
allTransitiveDependencies: null,
);
typedef PendingTodosCountRef = AutoDisposeProviderRef<int>;
String _$todosHash() => r'2ce152307a44fa5d6173831856732cfe2d082c36';
/// Todos State Provider - Fetches and manages todos list
///
/// Copied from [Todos].
@ProviderFor(Todos)
final todosProvider =
AutoDisposeAsyncNotifierProvider<Todos, List<Todo>>.internal(
Todos.new,
name: r'todosProvider',
debugGetCreateSourceHash:
const bool.fromEnvironment('dart.vm.product') ? null : _$todosHash,
dependencies: null,
allTransitiveDependencies: null,
);
typedef _$Todos = AutoDisposeAsyncNotifier<List<Todo>>;
// ignore_for_file: type=lint
// ignore_for_file: subtype_of_sealed_class, invalid_use_of_internal_member, invalid_use_of_visible_for_testing_member

View File

@@ -1,110 +1,34 @@
import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import '../providers/todo_providers.dart';
class HomeScreen extends StatefulWidget {
const HomeScreen({super.key});
class TodoScreen extends ConsumerStatefulWidget {
const TodoScreen({super.key});
@override
State<HomeScreen> createState() => _HomeScreenState();
ConsumerState<TodoScreen> createState() => _TodoScreenState();
}
class _HomeScreenState extends State<HomeScreen> {
bool _isLoading = false;
List<Map<String, dynamic>> _todos = [];
class _TodoScreenState extends ConsumerState<TodoScreen> {
String _searchQuery = '';
final TextEditingController _searchController = TextEditingController();
@override
void initState() {
super.initState();
_loadTodos();
}
@override
void dispose() {
_searchController.dispose();
super.dispose();
}
// Mock todos data - will be replaced with API call
Future<void> _loadTodos() async {
setState(() {
_isLoading = true;
});
// Simulate API call delay
await Future.delayed(const Duration(seconds: 1));
// Mock data simulating JSONPlaceholder response
final mockTodos = [
{
'id': 1,
'title': 'Complete project documentation',
'completed': false,
'userId': 1,
},
{
'id': 2,
'title': 'Review code changes',
'completed': true,
'userId': 1,
},
{
'id': 3,
'title': 'Update Flutter dependencies',
'completed': false,
'userId': 1,
},
{
'id': 4,
'title': 'Write unit tests',
'completed': false,
'userId': 2,
},
{
'id': 5,
'title': 'Fix navigation bug',
'completed': true,
'userId': 2,
},
];
if (mounted) {
setState(() {
_todos = mockTodos;
_isLoading = false;
});
}
}
List<Map<String, dynamic>> get _filteredTodos {
if (_searchQuery.isEmpty) {
return _todos;
}
return _todos.where((todo) {
return todo['title']
.toString()
.toLowerCase()
.contains(_searchQuery.toLowerCase());
}).toList();
}
void _toggleTodoStatus(int id) {
setState(() {
final todoIndex = _todos.indexWhere((todo) => todo['id'] == id);
if (todoIndex != -1) {
_todos[todoIndex]['completed'] = !_todos[todoIndex]['completed'];
}
});
}
Future<void> _refreshTodos() async {
await _loadTodos();
await ref.read(todosProvider.notifier).refresh();
}
@override
Widget build(BuildContext context) {
final theme = Theme.of(context);
final colorScheme = theme.colorScheme;
final todosAsync = ref.watch(todosProvider);
final filteredTodos = ref.watch(filteredTodosProvider(_searchQuery));
return Scaffold(
appBar: AppBar(
@@ -165,25 +89,61 @@ class _HomeScreenState extends State<HomeScreen> {
// Todos List
Expanded(
child: _isLoading
? const Center(
child: CircularProgressIndicator(),
)
: _todos.isEmpty
? _buildEmptyState()
: RefreshIndicator(
onRefresh: _refreshTodos,
child: _filteredTodos.isEmpty
? _buildNoResultsState()
: ListView.builder(
padding: const EdgeInsets.all(16.0),
itemCount: _filteredTodos.length,
itemBuilder: (context, index) {
final todo = _filteredTodos[index];
return _buildTodoCard(todo);
},
),
child: todosAsync.when(
data: (todos) {
if (todos.isEmpty) {
return _buildEmptyState();
}
if (filteredTodos.isEmpty) {
return _buildNoResultsState();
}
return RefreshIndicator(
onRefresh: _refreshTodos,
child: ListView.builder(
padding: const EdgeInsets.all(16.0),
itemCount: filteredTodos.length,
itemBuilder: (context, index) {
final todo = filteredTodos[index];
return _buildTodoCard(todo);
},
),
);
},
loading: () => const Center(child: CircularProgressIndicator()),
error: (error, stack) => Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
Icons.error_outline,
size: 64,
color: colorScheme.error,
),
const SizedBox(height: 16),
Text(
'Error loading todos',
style: theme.textTheme.headlineSmall?.copyWith(
color: colorScheme.error,
),
),
const SizedBox(height: 8),
Text(
error.toString(),
style: theme.textTheme.bodyMedium?.copyWith(
color: colorScheme.onSurfaceVariant,
),
textAlign: TextAlign.center,
),
const SizedBox(height: 16),
ElevatedButton.icon(
onPressed: _refreshTodos,
icon: const Icon(Icons.refresh),
label: const Text('Retry'),
),
],
),
),
),
),
],
),
@@ -202,10 +162,10 @@ class _HomeScreenState extends State<HomeScreen> {
);
}
Widget _buildTodoCard(Map<String, dynamic> todo) {
Widget _buildTodoCard(todo) {
final theme = Theme.of(context);
final colorScheme = theme.colorScheme;
final isCompleted = todo['completed'] as bool;
final isCompleted = todo.completed;
return Card(
margin: const EdgeInsets.only(bottom: 8.0),
@@ -220,13 +180,13 @@ class _HomeScreenState extends State<HomeScreen> {
),
leading: Checkbox(
value: isCompleted,
onChanged: (_) => _toggleTodoStatus(todo['id']),
onChanged: (_) => ref.read(todosProvider.notifier).toggleTodo(todo.id),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(4),
),
),
title: Text(
todo['title'],
todo.title,
style: theme.textTheme.bodyLarge?.copyWith(
decoration: isCompleted ? TextDecoration.lineThrough : null,
color: isCompleted
@@ -234,11 +194,27 @@ class _HomeScreenState extends State<HomeScreen> {
: colorScheme.onSurface,
),
),
subtitle: Text(
'ID: ${todo['id']} • User: ${todo['userId']}',
style: theme.textTheme.bodySmall?.copyWith(
color: colorScheme.onSurfaceVariant,
),
subtitle: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (todo.description != null && todo.description!.isNotEmpty)
Text(
todo.description!,
style: theme.textTheme.bodySmall?.copyWith(
color: colorScheme.onSurfaceVariant,
),
maxLines: 2,
overflow: TextOverflow.ellipsis,
),
const SizedBox(height: 4),
Text(
'ID: ${todo.id}',
style: theme.textTheme.bodySmall?.copyWith(
color: colorScheme.onSurfaceVariant.withOpacity(0.7),
fontSize: 11,
),
),
],
),
trailing: PopupMenuButton<String>(
onSelected: (value) {