invoice/lib/ui/expense/edit/expense_edit_notes.dart

137 lines
4.6 KiB
Dart

// Flutter imports:
import 'package:flutter/material.dart';
import 'package:flutter_redux/flutter_redux.dart';
import 'package:invoiceninja_flutter/redux/app/app_state.dart';
// Project imports:
import 'package:invoiceninja_flutter/constants.dart';
import 'package:invoiceninja_flutter/data/models/entities.dart';
import 'package:invoiceninja_flutter/redux/expense/expense_actions.dart';
import 'package:invoiceninja_flutter/ui/app/document_grid.dart';
import 'package:invoiceninja_flutter/ui/app/form_card.dart';
import 'package:invoiceninja_flutter/ui/app/forms/decorated_form_field.dart';
import 'package:invoiceninja_flutter/ui/app/help_text.dart';
import 'package:invoiceninja_flutter/ui/app/scrollable_listview.dart';
import 'package:invoiceninja_flutter/ui/expense/edit/expense_edit_vm.dart';
import 'package:invoiceninja_flutter/utils/completers.dart';
import 'package:invoiceninja_flutter/utils/localization.dart';
import 'package:invoiceninja_flutter/utils/platforms.dart';
class ExpenseEditNotes extends StatefulWidget {
const ExpenseEditNotes({
Key key,
@required this.viewModel,
}) : super(key: key);
final AbstractExpenseEditVM viewModel;
@override
ExpenseEditNotesState createState() => new ExpenseEditNotesState();
}
class ExpenseEditNotesState extends State<ExpenseEditNotes> {
final _publicNotesController = TextEditingController();
final _privateNotesController = TextEditingController();
List<TextEditingController> _controllers;
final _debouncer = Debouncer();
@override
void didChangeDependencies() {
_controllers = [
_publicNotesController,
_privateNotesController,
];
_controllers
.forEach((dynamic controller) => controller.removeListener(_onChanged));
final expense = widget.viewModel.expense;
_publicNotesController.text = expense.publicNotes;
_privateNotesController.text = expense.privateNotes;
_controllers
.forEach((dynamic controller) => controller.addListener(_onChanged));
super.didChangeDependencies();
}
@override
void dispose() {
_controllers.forEach((dynamic controller) {
controller.removeListener(_onChanged);
controller.dispose();
});
super.dispose();
}
void _onChanged() {
final viewModel = widget.viewModel;
final expense = viewModel.expense.rebuild((b) => b
..publicNotes = _publicNotesController.text.trim()
..privateNotes = _privateNotesController.text.trim());
if (expense != viewModel.expense) {
_debouncer.run(() {
viewModel.onChanged(expense);
});
}
}
@override
Widget build(BuildContext context) {
final store = StoreProvider.of<AppState>(context);
final localization = AppLocalization.of(context);
final viewModel = widget.viewModel;
final state = viewModel.state;
final expense = viewModel.expense;
final isFullscreen = state.prefState.isEditorFullScreen(EntityType.expense);
final showDocuments = isDesktop(context) && state.isEnterprisePlan;
return ScrollableListView(
children: <Widget>[
FormCard(
padding: isFullscreen
? const EdgeInsets.only(
left: kMobileDialogPadding / 2,
top: kMobileDialogPadding,
right: kMobileDialogPadding / 2,
)
: null,
children: <Widget>[
DecoratedFormField(
maxLines: showDocuments ? 6 : 10,
controller: _publicNotesController,
keyboardType: TextInputType.multiline,
label: localization.publicNotes,
),
DecoratedFormField(
maxLines: showDocuments ? 6 : 10,
controller: _privateNotesController,
keyboardType: TextInputType.multiline,
label: localization.privateNotes,
),
if (showDocuments)
if (expense.isNew || state.hasChanges())
SizedBox(
child: HelpText(localization.saveToUploadDocuments),
height: 200,
)
else
DocumentGrid(
documents: expense.documents.toList(),
onUploadDocument: (path) =>
widget.viewModel.onUploadDocument(context, path),
onDeleteDocument: (document, password, idToken) => widget
.viewModel
.onDeleteDocument(context, document, password, idToken),
onRenamedDocument: () =>
store.dispatch(LoadExpense(expenseId: expense.id)),
)
],
),
],
);
}
}