// Dart imports: import 'dart:async'; // Flutter imports: import 'package:flutter/material.dart'; // Package imports: import 'package:flutter_redux/flutter_redux.dart'; import 'package:flutter_styled_toast/flutter_styled_toast.dart'; import 'package:http/http.dart'; import 'package:redux/redux.dart'; // Project imports: import 'package:invoiceninja_flutter/data/models/models.dart'; import 'package:invoiceninja_flutter/redux/app/app_actions.dart'; import 'package:invoiceninja_flutter/redux/app/app_state.dart'; import 'package:invoiceninja_flutter/redux/project/project_actions.dart'; import 'package:invoiceninja_flutter/ui/app/dialogs/error_dialog.dart'; import 'package:invoiceninja_flutter/ui/project/view/project_view.dart'; import 'package:invoiceninja_flutter/utils/completers.dart'; import 'package:invoiceninja_flutter/utils/localization.dart'; class ProjectViewScreen extends StatelessWidget { const ProjectViewScreen({ Key? key, this.isFilter = false, }) : super(key: key); final bool isFilter; static const String route = '/project/view'; @override Widget build(BuildContext context) { return StoreConnector( converter: (Store store) { return ProjectViewVM.fromStore(store); }, builder: (context, vm) { return ProjectView( viewModel: vm, isFilter: isFilter, tabIndex: vm.state.projectUIState.tabIndex, ); }, ); } } class ProjectViewVM { ProjectViewVM({ required this.state, required this.project, required this.client, required this.company, required this.onEntityAction, required this.onEntityPressed, required this.onAddTaskPressed, required this.onRefreshed, required this.isSaving, required this.isLoading, required this.isDirty, required this.onUploadDocuments, }); factory ProjectViewVM.fromStore(Store store) { final state = store.state; final project = state.projectState.map[state.projectUIState.selectedId] ?? ProjectEntity(id: state.projectUIState.selectedId); final client = state.clientState.map[project.clientId] ?? ClientEntity(id: project.clientId); Future _handleRefresh(BuildContext context) { final completer = snackBarCompleter(AppLocalization.of(context)!.refreshComplete); store.dispatch(LoadProject(completer: completer, projectId: project.id)); return completer.future; } return ProjectViewVM( state: state, company: state.company, isSaving: state.isSaving, isLoading: state.isLoading, isDirty: project.isNew, project: project, client: client, onRefreshed: (context) => _handleRefresh(context), onEntityPressed: (BuildContext context, EntityType entityType, {bool? longPress = false}) { if (longPress == true && project.isActive && client.isActive) { handleProjectAction( context, [project], EntityAction.newEntityType(entityType)); } else { viewEntitiesByType( entityType: entityType, filterEntity: project, ); } }, onAddTaskPressed: (context) { createEntity( entity: TaskEntity(state: state).rebuild((b) => b ..projectId = project.id ..clientId = project.clientId), force: true); }, onEntityAction: (BuildContext context, EntityAction action) => handleEntitiesActions([project], action, autoPop: true), onUploadDocuments: (BuildContext context, List multipartFiles, bool isPrivate) { final completer = Completer>(); store.dispatch(SaveProjectDocumentRequest( isPrivate: isPrivate, multipartFile: multipartFiles, project: project, completer: completer)); completer.future.then((client) { showToast(AppLocalization.of(context)!.uploadedDocument); }).catchError((Object error) { showDialog( context: context, builder: (BuildContext context) { return ErrorDialog(error); }); }); }, ); } final AppState state; final ProjectEntity project; final ClientEntity client; final CompanyEntity? company; final Function(BuildContext, EntityAction) onEntityAction; final Function(BuildContext) onAddTaskPressed; final Function(BuildContext, EntityType, {bool? longPress}) onEntityPressed; final Function(BuildContext) onRefreshed; final bool isSaving; final bool isLoading; final bool isDirty; final Function(BuildContext, List, bool) onUploadDocuments; }