2 * Copyright © 2019 Damyan Ivanov.
3 * This file is part of MoLe.
4 * MoLe is free software: you can distribute it and/or modify it
5 * under the term of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your opinion), any later version.
9 * MoLe is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License terms for details.
14 * You should have received a copy of the GNU General Public License
15 * along with MoLe. If not, see <https://www.gnu.org/licenses/>.
18 package net.ktnx.mobileledger.ui.activity;
20 import android.content.Context;
21 import android.os.Bundle;
22 import android.renderscript.RSInvalidStateException;
23 import android.view.LayoutInflater;
24 import android.view.Menu;
25 import android.view.MenuInflater;
26 import android.view.View;
27 import android.view.ViewGroup;
29 import androidx.annotation.NonNull;
30 import androidx.annotation.Nullable;
31 import androidx.fragment.app.Fragment;
32 import androidx.fragment.app.FragmentActivity;
33 import androidx.lifecycle.ViewModelProvider;
34 import androidx.recyclerview.widget.ItemTouchHelper;
35 import androidx.recyclerview.widget.LinearLayoutManager;
36 import androidx.recyclerview.widget.RecyclerView;
38 import com.google.android.material.floatingactionbutton.FloatingActionButton;
39 import com.google.android.material.snackbar.Snackbar;
41 import net.ktnx.mobileledger.R;
42 import net.ktnx.mobileledger.model.Data;
43 import net.ktnx.mobileledger.model.LedgerTransaction;
44 import net.ktnx.mobileledger.model.LedgerTransactionAccount;
45 import net.ktnx.mobileledger.model.MobileLedgerProfile;
46 import net.ktnx.mobileledger.utils.Logger;
47 import net.ktnx.mobileledger.utils.Misc;
49 import org.jetbrains.annotations.NotNull;
51 import java.util.Date;
54 * A simple {@link Fragment} subclass.
55 * Activities that contain this fragment must implement the
56 * {@link OnNewTransactionFragmentInteractionListener} interface
57 * to handle interaction events.
59 public class NewTransactionFragment extends Fragment {
60 private NewTransactionItemsAdapter listAdapter;
61 private NewTransactionModel viewModel;
62 private RecyclerView list;
63 private FloatingActionButton fab;
64 private OnNewTransactionFragmentInteractionListener mListener;
65 private MobileLedgerProfile mProfile;
66 public NewTransactionFragment() {
67 // Required empty public constructor
68 setHasOptionsMenu(true);
71 public void onCreateOptionsMenu(@NonNull Menu menu, @NonNull MenuInflater inflater) {
72 super.onCreateOptionsMenu(menu, inflater);
73 inflater.inflate(R.menu.new_transaction_fragment, menu);
74 menu.findItem(R.id.action_reset_new_transaction_activity)
75 .setOnMenuItemClickListener(item -> {
81 public View onCreateView(LayoutInflater inflater, ViewGroup container,
82 Bundle savedInstanceState) {
83 // Inflate the layout for this fragment
84 return inflater.inflate(R.layout.fragment_new_transaction, container, false);
88 public void onActivityCreated(@Nullable Bundle savedInstanceState) {
89 super.onActivityCreated(savedInstanceState);
90 FragmentActivity activity = getActivity();
92 throw new RSInvalidStateException(
93 "getActivity() returned null within onActivityCreated()");
95 list = activity.findViewById(R.id.new_transaction_accounts);
96 viewModel = new ViewModelProvider(activity).get(NewTransactionModel.class);
97 mProfile = Data.profile.getValue();
98 listAdapter = new NewTransactionItemsAdapter(viewModel, mProfile);
99 list.setAdapter(listAdapter);
100 list.setLayoutManager(new LinearLayoutManager(activity));
101 Data.profile.observe(getViewLifecycleOwner(), profile -> {
103 listAdapter.setProfile(profile);
105 listAdapter.notifyDataSetChanged();
106 new ItemTouchHelper(new ItemTouchHelper.Callback() {
108 public int getMovementFlags(@NonNull RecyclerView recyclerView,
109 @NonNull RecyclerView.ViewHolder viewHolder) {
110 int flags = makeFlag(ItemTouchHelper.ACTION_STATE_IDLE, ItemTouchHelper.END);
111 // the top item is always there (date and description)
112 if (viewHolder.getAdapterPosition() > 0) {
113 if (viewModel.getAccountCount() > 2) {
114 flags |= makeFlag(ItemTouchHelper.ACTION_STATE_SWIPE,
115 ItemTouchHelper.START | ItemTouchHelper.END);
122 public boolean onMove(@NonNull RecyclerView recyclerView,
123 @NonNull RecyclerView.ViewHolder viewHolder,
124 @NonNull RecyclerView.ViewHolder target) {
128 public void onSwiped(@NonNull RecyclerView.ViewHolder viewHolder, int direction) {
129 if (viewModel.getAccountCount() == 2)
130 Snackbar.make(list, R.string.msg_at_least_two_accounts_are_required,
131 Snackbar.LENGTH_LONG)
132 .setAction("Action", null)
135 int pos = viewHolder.getAdapterPosition();
136 viewModel.removeItem(pos - 1);
137 listAdapter.notifyItemRemoved(pos);
138 viewModel.sendCountNotifications(); // needed after items re-arrangement
139 viewModel.checkTransactionSubmittable(listAdapter);
142 }).attachToRecyclerView(list);
144 viewModel.isSubmittable()
145 .observe(getViewLifecycleOwner(), isSubmittable -> {
149 fab.setEnabled(true);
158 viewModel.checkTransactionSubmittable(listAdapter);
160 fab = activity.findViewById(R.id.fab);
161 fab.setOnClickListener(v -> onFabPressed());
163 boolean keep = false;
165 Bundle args = getArguments();
167 String error = args.getString("error");
169 Logger.debug("new-trans-f", String.format("Got error: %s", error));
170 Snackbar.make(list, error, Snackbar.LENGTH_LONG)
177 if (savedInstanceState != null) {
178 keep |= savedInstanceState.getBoolean("keep", true);
179 focused = savedInstanceState.getInt("focused", 0);
185 viewModel.setFocusedItem(focused);
189 public void onSaveInstanceState(@NonNull Bundle outState) {
190 super.onSaveInstanceState(outState);
191 outState.putBoolean("keep", true);
192 final int focusedItem = viewModel.getFocusedItem();
193 outState.putInt("focused", focusedItem);
195 private void onFabPressed() {
196 fab.setEnabled(false);
197 Misc.hideSoftKeyboard(this);
198 if (mListener != null) {
199 Date date = viewModel.getDate();
200 LedgerTransaction tr =
201 new LedgerTransaction(null, date, viewModel.getDescription(), mProfile);
203 LedgerTransactionAccount emptyAmountAccount = null;
204 float emptyAmountAccountBalance = 0;
205 for (int i = 0; i < viewModel.getAccountCount(); i++) {
206 LedgerTransactionAccount acc =
207 new LedgerTransactionAccount(viewModel.getAccount(i));
208 if (acc.getAccountName()
213 if (acc.isAmountSet()) {
214 emptyAmountAccountBalance += acc.getAmount();
217 emptyAmountAccount = acc;
223 if (emptyAmountAccount != null)
224 emptyAmountAccount.setAmount(-emptyAmountAccountBalance);
226 mListener.onTransactionSave(tr);
231 public void onAttach(@NotNull Context context) {
232 super.onAttach(context);
233 if (context instanceof OnNewTransactionFragmentInteractionListener) {
234 mListener = (OnNewTransactionFragmentInteractionListener) context;
237 throw new RuntimeException(
238 context.toString() + " must implement OnFragmentInteractionListener");
243 public void onDetach() {
249 * This interface must be implemented by activities that contain this
250 * fragment to allow an interaction in this fragment to be communicated
251 * to the activity and potentially other fragments contained in that
254 * See the Android Training lesson <a href=
255 * "http://developer.android.com/training/basics/fragments/communicating.html"
256 * >Communicating with Other Fragments</a> for more information.
258 public interface OnNewTransactionFragmentInteractionListener {
259 void onTransactionSave(LedgerTransaction tr);