2026-05-07 16:28:01 +02:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
|
|
|
import 'package:flux/features/tickets/models/ticket_model.dart';
|
|
|
|
|
import 'package:flux/features/tickets/data/ticket_repository.dart';
|
|
|
|
|
import 'package:get_it/get_it.dart';
|
|
|
|
|
import 'ticket_list_state.dart';
|
|
|
|
|
|
|
|
|
|
class TicketListCubit extends Cubit<TicketListState> {
|
|
|
|
|
final TicketRepository _repository = GetIt.I.get<TicketRepository>();
|
|
|
|
|
static const int _limit = 20; // Paginazione a blocchi di 20
|
|
|
|
|
|
|
|
|
|
TicketListCubit() : super(const TicketListState()) {
|
2026-05-15 13:32:34 +02:00
|
|
|
loadTickets(refresh: true);
|
2026-05-07 16:28:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Recupera i ticket. Se reset = true, svuota la lista e riparte da offset 0.
|
2026-05-15 13:32:34 +02:00
|
|
|
Future<void> loadTickets({bool refresh = false}) async {
|
2026-05-07 16:28:01 +02:00
|
|
|
if (state.isLoading) return;
|
2026-05-15 13:32:34 +02:00
|
|
|
if (!refresh && state.hasReachedMax) return;
|
2026-05-07 16:28:01 +02:00
|
|
|
|
|
|
|
|
emit(
|
|
|
|
|
state.copyWith(
|
|
|
|
|
isLoading: true,
|
|
|
|
|
errorMessage: '',
|
2026-05-15 13:32:34 +02:00
|
|
|
tickets: refresh ? [] : state.tickets,
|
2026-05-07 16:28:01 +02:00
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
try {
|
2026-05-15 13:32:34 +02:00
|
|
|
final currentOffset = refresh ? 0 : state.tickets.length;
|
2026-05-07 16:28:01 +02:00
|
|
|
|
|
|
|
|
final newTickets = await _repository.fetchStoreTickets(
|
|
|
|
|
offset: currentOffset,
|
|
|
|
|
limit: _limit,
|
|
|
|
|
searchTerm: state.searchTerm,
|
|
|
|
|
dateRange: state.dateRange,
|
|
|
|
|
statusFilter: state.statusFilter,
|
|
|
|
|
ticketTypeFilter: state.ticketTypeFilter,
|
|
|
|
|
staffIdFilter: state.staffIdFilter,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
emit(
|
|
|
|
|
state.copyWith(
|
2026-05-15 13:32:34 +02:00
|
|
|
tickets: refresh ? newTickets : [...state.tickets, ...newTickets],
|
2026-05-07 16:28:01 +02:00
|
|
|
isLoading: false,
|
|
|
|
|
hasReachedMax: newTickets.length < _limit,
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
emit(state.copyWith(isLoading: false, errorMessage: e.toString()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Aggiorna i filtri e ricarica tutto da zero
|
|
|
|
|
void updateFilters({
|
|
|
|
|
String? searchTerm,
|
|
|
|
|
DateTimeRange? dateRange,
|
|
|
|
|
TicketStatus? statusFilter,
|
|
|
|
|
TicketType? ticketTypeFilter,
|
|
|
|
|
String? staffIdFilter,
|
|
|
|
|
bool clearSearch = false,
|
|
|
|
|
bool clearDate = false,
|
|
|
|
|
bool clearStatus = false,
|
|
|
|
|
}) {
|
|
|
|
|
emit(
|
|
|
|
|
state.copyWith(
|
|
|
|
|
searchTerm: searchTerm,
|
|
|
|
|
dateRange: dateRange,
|
|
|
|
|
statusFilter: statusFilter,
|
|
|
|
|
ticketTypeFilter: ticketTypeFilter,
|
|
|
|
|
staffIdFilter: staffIdFilter,
|
|
|
|
|
clearSearch: clearSearch,
|
|
|
|
|
clearDate: clearDate,
|
|
|
|
|
clearStatus: clearStatus,
|
|
|
|
|
),
|
|
|
|
|
);
|
2026-05-15 13:32:34 +02:00
|
|
|
loadTickets(refresh: true); // Applica i filtri e ricarica
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-16 11:51:26 +02:00
|
|
|
void toggleTicketSelection(TicketModel ticket) {
|
|
|
|
|
final currentSelection = Set<TicketModel>.from(state.selectedTickets);
|
|
|
|
|
if (currentSelection.contains(ticket)) {
|
|
|
|
|
currentSelection.remove(ticket);
|
2026-05-15 13:32:34 +02:00
|
|
|
} else {
|
2026-05-16 11:51:26 +02:00
|
|
|
currentSelection.add(ticket);
|
2026-05-15 13:32:34 +02:00
|
|
|
}
|
2026-05-16 11:51:26 +02:00
|
|
|
emit(state.copyWith(selectedTickets: currentSelection));
|
2026-05-15 13:32:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void clearSelection() {
|
2026-05-16 11:51:26 +02:00
|
|
|
emit(state.copyWith(selectedTickets: {}));
|
2026-05-15 13:32:34 +02:00
|
|
|
}
|
|
|
|
|
|
2026-05-16 11:51:26 +02:00
|
|
|
void selectAll(List<TicketModel> tickets) {
|
|
|
|
|
emit(state.copyWith(selectedTickets: tickets.toSet()));
|
2026-05-07 16:28:01 +02:00
|
|
|
}
|
|
|
|
|
}
|